using Aspose.Words; using AutoMapper; using EyeSoft.Extensions; using Microsoft.Extensions.Logging; using Newtonsoft.Json; using NPOI.SS.Formula.Functions; using OASystem.Domain; using OASystem.Domain.Dtos.Groups; using OASystem.Domain.Entities.Groups; using OASystem.Domain.Entities.Resource; using OASystem.Domain.ViewModels.Groups; using OASystem.Domain.ViewModels.Statistics; using Org.BouncyCastle.Asn1.X500; using System; using System.Diagnostics; using System.Drawing; using System.Reflection; using System.Runtime.Intrinsics.Arm; using System.Text.RegularExpressions; namespace OASystem.Infrastructure.Repositories.Groups { /// /// 团组流程总览表仓储 /// public class ProcessOverviewRepository : BaseRepository { private readonly IMapper _mapper; private readonly DelegationInfoRepository _groupRep; private readonly ILogger _logger; public ProcessOverviewRepository(SqlSugarClient sqlSugar, IMapper mapper, ILogger logger, DelegationInfoRepository groupRep) : base(sqlSugar) { _mapper = mapper; _groupRep = groupRep; _logger = logger; } #region 团组流程 /// /// 基础数据初始化-团组流程 /// /// /// /// public async Task> ProcessDataInitAsync(int groupId, int currUserId, List visaCountries) { var processs = new List(); // 团组验证 var groupInfo = await _sqlSugar.Queryable().FirstAsync(g => g.Id == groupId); if (groupInfo == null) return processs; // 流程验证 var existingProcesses = await _sqlSugar.Queryable() .Where(p => p.IsDel == 0 && p.GroupId == groupId) .ToListAsync(); if (existingProcesses.Any()) return processs; //基础数据 var users = await GetUersAsync(); var custInfo = _sqlSugar.Queryable() .Where(c => c.DiId == groupId && c.IsDel == 0) .OrderByDescending(c => c.CreateTime) .First(); var airTripCodeInfo = _sqlSugar.Queryable() .Where(x => x.IsDel == 0 && x.DiId == groupId) .OrderByDescending(x => x.CreateTime) .First(); var backListInfo = _sqlSugar.Queryable().Where(x => x.DiId == groupId && x.IsDel == 0).First(); #region 商邀报批流程 var oa_proc = GroupProcessType.Invitation; var oaNode1 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, oa_proc, 1); var oaNode2 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, oa_proc, 2); var oaNode3 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, oa_proc, 3); var oaNode4 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, oa_proc, 4); var oaNode5 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, oa_proc, 5); var oaNode6 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, oa_proc, 6); var oaNode7 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, oa_proc, 7); processs.Add( Grp_ProcessOverview.Create(groupId, 1, oa_proc, ProcessStatus.InProgress, currUserId, new List() { Grp_ProcessNode.Create(1, oaNode1.NodeName,oaNode1.NodeTips,ProcessStatus.InProgress, true,false,false,false,currUserId,oaNode1.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(2, oaNode2.NodeName,oaNode2.NodeTips, ProcessStatus.InProgress, false,false,false,false,currUserId,oaNode2.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(3, oaNode3.NodeName,oaNode3.NodeTips,ProcessStatus.InProgress, false,false,false,false, currUserId ,oaNode3.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(4, oaNode4.NodeName,oaNode4.NodeTips,ProcessStatus.InProgress, false,false,false,false, currUserId,oaNode4.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(5, oaNode5.NodeName,oaNode5.NodeTips,ProcessStatus.InProgress, false,false,false,true, currUserId,oaNode5.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(6, oaNode6.NodeName,oaNode6.NodeTips,ProcessStatus.InProgress, false,false,false,true, currUserId,oaNode6.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(7, oaNode7.NodeName,oaNode7.NodeTips,ProcessStatus.InProgress, false,false,true,false, currUserId,oaNode7.PromptPerson.Select(x => x.Id).ToList()), } )); #endregion #region 签证流程 var visa_proc = GroupProcessType.Visa; //单独处理签证流程节点 var visaNodes = new List(); if (visaCountries != null && visaCountries.Count > 0) { var visaDefualtNodes = new List(); for (int i = 1; i < visaCountries.Count + 1; i++) { visaDefualtNodes.Add(VisaProcessNode.Info(i, visaCountries[i - 1].ToString())); } var visaNode1 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, visa_proc, 1); var visaNode2 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, visa_proc, 2); visaNodes.Add(Grp_ProcessNode.Create(1, visaNode1.NodeName, visaNode1.NodeTips, ProcessStatus.InProgress, true, false, false, false, currUserId, visaNode1.PromptPerson.Select(x => x.Id).ToList(), JsonConvert.SerializeObject(visaDefualtNodes))); visaNodes.Add(Grp_ProcessNode.Create(2, visaNode2.NodeName, visaNode2.NodeTips, ProcessStatus.InProgress, false, false, true, false, currUserId, visaNode2.PromptPerson.Select(x => x.Id).ToList())); } processs.Add(Grp_ProcessOverview.Create(groupId, 2, visa_proc, ProcessStatus.UnStarted, currUserId, visaNodes)); #endregion #region 机票流程 var air_proc = GroupProcessType.AirTicket; var airNode1 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, air_proc, 1); var airNode2 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, air_proc, 2); var airNode3 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, air_proc, 3); var airNode4 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, air_proc, 4); var airNode5 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, air_proc, 5); var airNode6 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, air_proc, 6); var airNode7 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, air_proc, 7); var airNode8 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, air_proc, 8); processs.Add( Grp_ProcessOverview.Create(groupId, 3, air_proc, ProcessStatus.InProgress, currUserId, new List() { Grp_ProcessNode.Create(1, airNode1.NodeName, airNode1.NodeTips, ProcessStatus.InProgress, true,false,false,false, currUserId ,airNode1.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(2, airNode2.NodeName, airNode2.NodeTips, ProcessStatus.InProgress, false,false,false,false,currUserId ,airNode2.PromptPerson.Select(x => x.Id).ToList() ), Grp_ProcessNode.Create(3, airNode3.NodeName, airNode3.NodeTips, ProcessStatus.InProgress, false,false,false,false, currUserId ,airNode3.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(4, airNode4.NodeName, airNode4.NodeTips, ProcessStatus.InProgress, false,false,false,false, currUserId ,airNode4.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(5, airNode5.NodeName, airNode5.NodeTips, ProcessStatus.InProgress, false,false,false,false, currUserId ,airNode5.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(6, airNode6.NodeName, airNode6.NodeTips, ProcessStatus.InProgress, false,false,false,false,currUserId ,airNode6.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(7, airNode7.NodeName, airNode7.NodeTips, ProcessStatus.InProgress, false,false,true,false, currUserId ,airNode7.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(8, airNode8.NodeName, airNode8.NodeTips, ProcessStatus.InProgress, false,false,true,false, currUserId ,airNode8.PromptPerson.Select(x => x.Id).ToList()) } ) ); #endregion #region 酒店流程 var hotel_proc = GroupProcessType.Hotel; var hotelNode1 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, hotel_proc, 1); var hotelNode2 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, hotel_proc, 2); var hotelNode3 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, hotel_proc, 3); var hotelNode4 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, hotel_proc, 4); var hotelNode5 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, hotel_proc, 5); processs.Add( Grp_ProcessOverview.Create(groupId, 4, hotel_proc, ProcessStatus.InProgress, currUserId, new List() { Grp_ProcessNode.Create(1, hotelNode1.NodeName, hotelNode1.NodeTips, ProcessStatus.InProgress, true, false, false, false, currUserId, hotelNode1.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(2, hotelNode2.NodeName, hotelNode2.NodeTips, ProcessStatus.InProgress, false, false, false,false, currUserId, hotelNode2.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(3, hotelNode3.NodeName, hotelNode3.NodeTips, ProcessStatus.InProgress,false, false, false,false, currUserId, hotelNode3.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(4, hotelNode4.NodeName, hotelNode4.NodeTips, ProcessStatus.InProgress, false, false, false,false,currUserId, hotelNode4.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(5, hotelNode5.NodeName, hotelNode5.NodeTips, ProcessStatus.InProgress, false, false, true,false, currUserId, hotelNode5.PromptPerson.Select(x => x.Id).ToList()), } ) ); #endregion #region 地接流程 var op_proc = GroupProcessType.LocalGuide; var opNode1 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, op_proc, 1); var opNode2 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, op_proc, 2); var opNode3 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, op_proc, 3); var opNode4 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, op_proc, 4); var opNode5 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, op_proc, 5); var opNode6 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, op_proc, 6); var opNode7 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, op_proc, 7); processs.Add( Grp_ProcessOverview.Create(groupId, 5, op_proc, ProcessStatus.InProgress, currUserId, new List() { Grp_ProcessNode.Create(1, opNode1.NodeName, opNode1.NodeTips, ProcessStatus.InProgress, true, false, false,false, currUserId, opNode1.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(2, opNode2.NodeName, opNode2.NodeTips, ProcessStatus.InProgress, false, false, false,false, currUserId, opNode2.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(3, opNode3.NodeName, opNode3.NodeTips, ProcessStatus.InProgress, false, false, false, false, currUserId, opNode3.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(4, opNode4.NodeName, opNode4.NodeTips, ProcessStatus.InProgress, false, false, false,false, currUserId, opNode4.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(5, opNode5.NodeName, opNode5.NodeTips, ProcessStatus.InProgress, false, false, false,false, currUserId, opNode5.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(6, opNode6.NodeName, opNode6.NodeTips, ProcessStatus.InProgress, false, false, false,false, currUserId, opNode6.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(7, opNode7.NodeName, opNode7.NodeTips, ProcessStatus.InProgress, false, false, true, false, currUserId, opNode7.PromptPerson.Select(x => x.Id).ToList()) } ) ); #endregion #region 费用结算流程 var fee_proc = GroupProcessType.FeeSettle; var feeNode1 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, fee_proc, 1); var feeNode2 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, fee_proc, 2); var feeNode3 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, fee_proc, 3); var feeNode4 = GetNodeDetailsByProcAndNode(groupInfo, users, custInfo, airTripCodeInfo, backListInfo, fee_proc, 4); processs.Add( Grp_ProcessOverview.Create(groupId, 6, fee_proc, ProcessStatus.InProgress, currUserId, new List() { Grp_ProcessNode.Create(1, feeNode1.NodeName, feeNode1.NodeTips, ProcessStatus.InProgress, true, true, false,false,currUserId, feeNode1.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(2, feeNode2.NodeName, feeNode2.NodeTips, ProcessStatus.InProgress, false, false, false,false,currUserId, feeNode2.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(3, feeNode3.NodeName, feeNode3.NodeTips, ProcessStatus.InProgress, false, false, false,false,currUserId, feeNode3.PromptPerson.Select(x => x.Id).ToList()), Grp_ProcessNode.Create(4, feeNode4.NodeName, feeNode4.NodeTips, ProcessStatus.InProgress, false, false, false,false, currUserId , feeNode4.PromptPerson.Select(x => x.Id).ToList()), } ) ); #endregion return processs; } /// /// 团组流程初始化 /// /// 创建流程请求参数 /// 创建的流程信息 public async Task ProcessInitAsync(int groupId, int currUserId) { //团组验证 var groupInfo = await _sqlSugar.Queryable().FirstAsync(g => g.Id == groupId); if (groupInfo == null) { return new Result { Code = 400, Msg = "团组不存在" }; } // 检查是否已存在流程 var existingProcesses = await _sqlSugar.Queryable() .Where(p => p.IsDel == 0 && p.GroupId == groupId) .ToListAsync(); if (existingProcesses.Any()) { return new Result { Code = 400, Msg = "该团组的流程已存在" }; } //处理签证国家 var visaCountries = _groupRep.GroupSplitCountry(groupInfo.VisitCountry); // 定义默认的流程节点 var processs = await ProcessDataInitAsync(groupId, currUserId, visaCountries); _sqlSugar.BeginTran(); foreach (var item in processs) { var processId = await _sqlSugar.Insertable(item).ExecuteReturnIdentityAsync(); if (processId < 1) { _sqlSugar.RollbackTran(); return new Result { Code = 400, Msg = "团组流程进度总览表添加失败!" }; } item.Id = processId; // 记录流程日志 await LogProcessOpAsync(null, item, "Create", currUserId); var nodes = item.Nodes.Select((nodeDto, index) => new Grp_ProcessNode { ProcessId = processId, NodeName = nodeDto.NodeName, NodeOrder = nodeDto.NodeOrder, OverallStatus = nodeDto.OverallStatus, NodeDescTips = nodeDto.NodeDescTips, OpUserList = nodeDto.OpUserList, //Country = nodeDto.Country, IsCurrent = nodeDto.IsCurrent, IsAssist = nodeDto.IsAssist, IsPart = nodeDto.IsPart, IsFileUp = nodeDto.IsFileUp, Remark = nodeDto.Remark }).ToList(); var nodeIds = await _sqlSugar.Insertable(nodes).ExecuteCommandAsync(); if (nodeIds < 1) { _sqlSugar.RollbackTran(); return new Result { Code = 400, Msg = "团组流程进度流程节点添加失败!" }; } //设置节点ID nodes = await _sqlSugar.Queryable().Where(x => x.IsDel == 0 && x.ProcessId == processId).ToListAsync(); //记录节点日志 foreach (var node in nodes) { await LogNodeOpAsync(null, node, "Create", currUserId); } } _sqlSugar.CommitTran(); return new Result { Code = 200, Msg = "添加成功!" }; ; } /// /// 获取团组的所有流程及节点详情 /// /// 创建流程请求参数 /// 创建的流程信息 public async Task ProcessesDetailsAsync(int groupId) { //团组验证 var groupInfo = await _sqlSugar.Queryable().FirstAsync(g => g.Id == groupId); if (groupInfo == null) { return new Result { Code = 400, Msg = "团组不存在" }; } // 检查是否已存在流程 var existingProcesses = await _sqlSugar.Queryable().Where(p => p.IsDel == 0 && p.GroupId == groupId).ToListAsync(); if (!existingProcesses.Any()) { //新建团组流程 var res = await ProcessInitAsync(groupId, 4); if (res.Code != 200) { return res; } } //基础数据 var opUsers = await GetUersAsync(); var custInfo = _sqlSugar.Queryable() .Where(c => c.DiId == groupId && c.IsDel == 0) .OrderByDescending(c => c.CreateTime) .First(); var airTripCodeInfo = _sqlSugar.Queryable() .Where(x => x.IsDel == 0 && x.DiId == groupId) .OrderByDescending(x => x.CreateTime) .First(); var backListInfo = _sqlSugar.Queryable().Where(x => x.DiId == groupId && x.IsDel == 0).First(); var users = await _sqlSugar.Queryable().Select(x => new { x.Id, x.CnName }).ToListAsync(); var processData = await _sqlSugar.Queryable() .Where(p => p.GroupId == groupId && p.IsDel == 0) .Mapper( p => p.Nodes, p => p.Nodes.First().ProcessId ) .ToListAsync(); // 预先构建用户字典,提升查询性能 var userDict = users.ToDictionary(u => u.Id, u => u.CnName); var processes = processData.Select(p => { var orderedNodes = p.Nodes.Where(x => x.IsDel == 0).OrderBy(n => n.NodeOrder).ToList(); var totalNodes = orderedNodes.Count; return new ProcessDetailsView() { Id = p.Id, GroupId = p.GroupId, ProcessType = p.ProcessType, ProcessName = p.ProcessType.GetEnumDescription(), Nodes = orderedNodes.Select((n, index) => { var isLastNode = index == totalNodes - 1; var isSecondLastNode = index == totalNodes - 2; var isFifthStep = index == 4; // 计算按钮状态 bool isEnaAssistBtn = p.ProcessType == GroupProcessType.FeeSettle && n.NodeOrder == 1; // 文件上传按钮启用规则 bool isEnaFileUpBtn = false; // 是否参与按钮启用 bool isEnaPartBtn = false; // 规则1:商邀流程第五步启用参与按钮 if (p.ProcessType == GroupProcessType.Invitation && isFifthStep) { isEnaPartBtn = true; } // 规则2:机票流程倒数第二步启用上传按钮 else if (p.ProcessType == GroupProcessType.AirTicket && isSecondLastNode) { isEnaFileUpBtn = true; } // 规则3:默认流程节点最后一步启用上传按钮 else if (isLastNode && p.ProcessType != GroupProcessType.FeeSettle) { isEnaFileUpBtn = true; } // 处理签证子节点 List visaSubNodes = new(); if (p.ProcessType == GroupProcessType.Visa && n.NodeOrder == 1) { visaSubNodes = JsonConvert.DeserializeObject>(n.Remark ?? "[]") ?? new List(); } // 获取操作人姓名 string operatorName = "-"; if (n.Operator.HasValue && userDict.TryGetValue(n.Operator.Value, out var name)) { operatorName = name; } string nodeTipsMsg = GetNodeDetailsByProcAndNode(groupInfo, opUsers, custInfo, airTripCodeInfo, backListInfo, p.ProcessType, n.NodeOrder)?.NodeTips; return new ProcessNodeDetailsView() { Id = n.Id, ProcessId = n.ProcessId, NodeOrder = n.NodeOrder, NodeName = n.NodeName, OverallStatus = n.OverallStatus, StatusText = n.OverallStatus.GetEnumDescription(), Operator = operatorName, OpeateTime = n.OperationTime?.ToString("yyyy-MM-dd HH:mm:ss") ?? "-", ActualDone = n.ActualDone?.ToString("yyyy-MM-dd HH:mm:ss") ?? "", OpUserList = n.OpUserList, //可操作用户列表 NodeDescTips = nodeTipsMsg, IsEnaAssistBtn = isEnaAssistBtn, // 是否启用财务流程首节点协助按钮 IsAssist = n.IsAssist, // 财务流程首节点 存储值 IsEnaFileUpBtn = isEnaFileUpBtn, // 是否启用上传文件按钮 IsFileUp = n.IsFileUp, // 上传文件节点 存储值 IsEnaPartBtn = isEnaPartBtn, // 是否启用参与按钮 IsPart = n.IsPart, // 参与按钮 存储值 VisaSubNodes = visaSubNodes, // 签证节点类型使用 Remark = n.Remark }; }).ToList() }; }).ToList(); return new Result { Code = 200, Data = processes, Msg = "查询成功!" }; } /// /// 指定日期增加工作日 /// /// /// /// private static DateTime AddWeekdays(DateTime startDate, int days) { DateTime currentDate = startDate; if (days >= 0) { while (days > 0) { currentDate = currentDate.AddDays(1); DayOfWeek day = currentDate.DayOfWeek; if (day != DayOfWeek.Saturday && day != DayOfWeek.Sunday) { days--; } } } else { days = Math.Abs(days); while (days > 0) { currentDate = currentDate.AddDays(-1); DayOfWeek day = currentDate.DayOfWeek; if (day != DayOfWeek.Saturday && day != DayOfWeek.Sunday) { days--; } } } return currentDate; } /// /// 团组流程单节点详情 /// /// /// /// /// /// /// public static GroupProcFullNodeDetails GetNodeDetailsByProcAndNode( Grp_DelegationInfo groupInfo, List users, Grp_TourClientList custInfo, Air_TicketBlackCode airTripCodeInfo, Grp_InvertedList backListInfo, GroupProcessType procType, int nodeOrder ) { return GetNodeDetails( groupInfo, users, custInfo, airTripCodeInfo, backListInfo ).FirstOrDefault(x => x.ProcType == procType && x.NodeOrder == nodeOrder); } /// /// 团组流程节点详情计算 /// /// /// /// /// /// /// public static List GetNodeDetails( Grp_DelegationInfo groupInfo, List users, Grp_TourClientList custInfo, Air_TicketBlackCode airTripCodeInfo, Grp_InvertedList backListInfo ) { var nodeDetails = new List(); int groupId = groupInfo.Id; int groupType = groupInfo.TeamDid; string groupName = groupInfo.TeamName; groupInfo.VisitDate = groupInfo.VisitDate.AddDays(1); //第二天开始计算 #region 商邀 var oa_proc = GroupProcessType.Invitation; //用户可操作权限 var oa_users = NodeOpUserTpl(users)[oa_proc]; //1. 完成基础请示、报批日程初稿”,4个工作日(仍需根据客户意见和联系情况及时修改补充所需其他材料,例如印证文件、成果表等,直至终稿) DateTime? xy_timeBase1 = null; if (groupInfo.Step == 1 || groupInfo.Step == 2) { if (groupInfo.StepOperationTime.HasValue) xy_timeBase1 = groupInfo.StepOperationTime.Value; } nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, oa_proc, 1, xy_timeBase1, 4, true, oa_users) ); //2. 7个工作日,所有报批机构前部联系,邀请机构一个国家不少于4家进行重点对接(4家机构中,其中3家机构需有效对接,其中1家可为付费机构备选)) DateTime? xy_timeBase2 = groupInfo.VisitDate; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, oa_proc, 2, xy_timeBase2, 7, true, oa_users) ); //3. 10个工作日,根据最新情况,联系公务机构1/3取得回应;邀请机构基本明确。 nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, oa_proc, 3, null, 0, true, oa_users) ); //4. 正式名单下放后2周内(含非工作日)。如团组前期准备时间已经较长,则按客户要求尽快提供。 加急团组备注特殊情况。 DateTime? xy_timeBase4 = null; if (custInfo != null) xy_timeBase4 = custInfo.CreateTime; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, oa_proc, 4, xy_timeBase4, 14, false, oa_users) ); //5. 团组出发前,5个工作日完成所有公务确认工作。 DateTime? xy_timeBase5 = groupInfo.VisitDate.AddDays(-1); //时间倒推,调整回原始出发日期 nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, oa_proc, 5, xy_timeBase5, -5, true, oa_users) ); //6. nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, oa_proc, 6, null, 0, true, oa_users) ); //7. 如果需要上传请在团组结束前完成 DateTime? xy_timeBase7 = groupInfo.VisitEndDate; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, oa_proc, 7, xy_timeBase7, 0, true, oa_users) ); #endregion #region 签证 var visa_proc = GroupProcessType.Visa; //用户可操作权限 var visa_users = NodeOpUserTpl(users)[visa_proc]; //1. nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, visa_proc, 1, null, 0, true, visa_users) ); //2. 按进度实际签证办理落实情况,团组出发前上传票据。 DateTime? qz_timeBase2 = groupInfo.VisitDate; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, visa_proc, 2, qz_timeBase2, 7, true, visa_users) ); #endregion #region 机票 var air_proc = GroupProcessType.AirTicket; //用户可操作权限 var air_users = NodeOpUserTpl(users)[air_proc]; //1. 建团后打勾确认出团的时候开始24小时内。 DateTime? air_timeBase1 = null; if (groupInfo.Step == 1 || groupInfo.Step == 2) { if (groupInfo.StepOperationTime.HasValue) air_timeBase1 = groupInfo.StepOperationTime.Value; } nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, air_proc, 1, air_timeBase1, 0, true, air_users) ); //2. nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, air_proc, 2, null, 0, true, visa_users) ); //3. 完成机票采购确认(含预算核对、出票确认等) nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, air_proc, 3, null, 0, true, visa_users) ); //4. nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, air_proc, 4, null, 0, true, visa_users) ); //5. 团组出发前2个工作日 DateTime? air_timeBase5 = groupInfo.VisitDate; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, air_proc, 5, air_timeBase5, -2, true, visa_users) ); //6. nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, air_proc, 6, null, 2, true, visa_users) ); //7. 机票蓝联打票及上传机票超支费用账单,团组归国后5个工作日内 DateTime? air_timeBase7 = groupInfo.VisitEndDate; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, air_proc, 7, air_timeBase7, 5, true, visa_users) ); //8. 团组归国后10个工作日内 *按机票报价*0.999折扣出具机票报销蓝联、行程单及机票说明 DateTime? air_timeBase8 = groupInfo.VisitEndDate; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, air_proc, 8, air_timeBase8, 10, true, visa_users) ); #endregion #region 酒店 var hotel_proc = GroupProcessType.Hotel; //用户可操作权限 var hotel_users = NodeOpUserTpl(users)[hotel_proc]; //1. 筛选并按照预算标准,对目标酒店进行询价、比价、谈价 \r\n2. 建团后打勾确认出团的时候开始2个工作日。 DateTime? hotel_timeBase1 = null; if (groupInfo.Step == 1 || groupInfo.Step == 2) { if (groupInfo.StepOperationTime.HasValue) hotel_timeBase1 = groupInfo.StepOperationTime.Value; } nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, hotel_proc, 1, hotel_timeBase1, 2, true, hotel_users) ); //2. nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, hotel_proc, 2, null, 0, true, hotel_users) ); //3. nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, hotel_proc, 3, null, 0, true, hotel_users) ); //4. 团组出发前2个工作日 DateTime? hotel_timeBase4 = groupInfo.VisitDate; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, hotel_proc, 4, hotel_timeBase4, -2, true, hotel_users) ); //5. 团组结束后5个工作日内 DateTime? hotel_timeBase5 = groupInfo.VisitEndDate; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, hotel_proc, 5, hotel_timeBase5, 5, true, hotel_users) ); #endregion #region 地接 var localGuide_proc = GroupProcessType.LocalGuide; //用户可操作权限 var localGuide_users = NodeOpUserTpl(users)[localGuide_proc]; //1. 机票行程代码最后一段录入后1个工作日内。 DateTime? dj_timeBase1 = null; if (airTripCodeInfo != null) dj_timeBase1 = airTripCodeInfo.CreateTime; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, localGuide_proc, 1, dj_timeBase1, 1, true, localGuide_users) ); //2. 团组出行前20个工作日 DateTime? dj_timeBase2 = groupInfo.VisitDate; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, localGuide_proc, 2, dj_timeBase2, -20, true, localGuide_users) ); //3. 上一步往后3个工作日内 DateTime? dj_timeBase3 = groupInfo.VisitDate; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, localGuide_proc, 3, dj_timeBase3, -17, true, localGuide_users) ); //4. 上一步往后2个工作日内 DateTime? dj_timeBase4 = groupInfo.VisitDate; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, localGuide_proc, 4, dj_timeBase4, -15, true, localGuide_users) ); //5. 1.制定最终《行程单》及《出行手册》 \r\n2. 倒推表里开行前会 -3天。 DateTime? dj_timeBase5 = null; if (backListInfo != null) { if (DateTime.TryParse(backListInfo.PreTripMeetingDt, out DateTime dateTime)) { dj_timeBase5 = dateTime; } } nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, localGuide_proc, 5, dj_timeBase5, -3, true, localGuide_users) ); //6. nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, localGuide_proc, 6, null, 0, true, localGuide_users) ); //7. 团组归国后5个工作日内) *上传最终报批行程,确定城市间交通最终版报价分配;地接账单(清楚标注超时及其他项超支费用)、地接交通费用原始票据、城市间交通明细表; DateTime? dj_timeBase7 = groupInfo.VisitEndDate; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, localGuide_proc, 7, dj_timeBase7, 5, true, localGuide_users) ); #endregion #region 费用计算 var feeSettle_proc = GroupProcessType.FeeSettle; //用户可操作权限 var feeSettle_users = NodeOpUserTpl(users)[feeSettle_proc]; //1. nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, feeSettle_proc, 1, null, 0, true, feeSettle_users) ); //2. nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, feeSettle_proc, 2, null, 0, true, feeSettle_users) ); //3. 团组归国后12个工作日内 DateTime? fyjs_timeBase3 = groupInfo.VisitEndDate; nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, feeSettle_proc, 3, fyjs_timeBase3, 12, true, feeSettle_users) ); //4. nodeDetails.Add( GroupProcFullNodeDetails.Create(groupId, groupName, groupType, feeSettle_proc, 4, null, 0, true, feeSettle_users) ); #endregion return nodeDetails; } #region 计算所有团时间节点基础数据 /// /// 团组流程 - 数据初始化 ALL /// /// /// /// public async Task> GetGroupAllProcessNodeInfoAsync() { var processs = new List(); var groupTypeIds = new List() { 38, // 政府团 39, // 企业团 40, // 散客团 1048, // 高校团 248, // 非团组 }; var groupInfos = await _sqlSugar.Queryable().Where(x => x.IsDel == 0 && groupTypeIds.Contains(x.TeamDid)).ToListAsync(); if (groupInfos.Any()) { var groupIds = groupInfos.Select(x => x.Id).ToList(); //基础数据 var opUsers = await GetUersAsync(); var custInfos = await _sqlSugar.Queryable() .Where(x => x.IsDel == 0 && groupIds.Contains( x.DiId)) .OrderByDescending(c => c.CreateTime) .ToListAsync(); var airTripCodeInfos = await _sqlSugar.Queryable() .Where(x => x.IsDel == 0 && groupIds.Contains(x.DiId)) .OrderByDescending(x => x.CreateTime) .ToListAsync(); var backListInfos = await _sqlSugar.Queryable().Where(x => x.IsDel == 0 && groupIds.Contains(x.DiId)).ToListAsync(); foreach (var groupInfo in groupInfos) { var custInfo = custInfos.FirstOrDefault(x => x.DiId == groupInfo.Id); var airTripCodeInfo = airTripCodeInfos.FirstOrDefault(x => x.DiId == groupInfo.Id); var backListInfo = backListInfos.FirstOrDefault(x => x.DiId == groupInfo.Id); var groupNodeDetails = GetNodeDetails( groupInfo, opUsers, custInfo, airTripCodeInfo, backListInfo ); processs.AddRange(groupNodeDetails); } } return processs; } /// /// 获取用户信息 /// /// public async Task> GetUersAsync() { return await _sqlSugar.Queryable() .LeftJoin((u, c) => u.CompanyId == c.Id) .LeftJoin((u, c, d) => u.DepId == d.Id) .LeftJoin((u, c, d, jp) => u.JobPostId == jp.Id) .Where((u, c, d, jp) => u.IsDel == 0) .Select((u, c, d, jp) => new NodeOpUserInfo() { Id = u.Id, CnName = u.CnName, CompanyId = u.CompanyId, CompanyName = c.CompanyName, DepId = u.DepId, DepName = d.DepName, JobPostId = u.JobPostId, JobName = jp.JobName, QiyeChatUserId = u.QiyeChatUserId, }) .ToListAsync(); } /// /// 节点可操作人员 /// /// public static Dictionary> NodeOpUserTpl(List users) { var result = new Dictionary>(); var defaultJobNames = new List() { "OP主管" }; #region 商邀报批流程 //节点可操作用户列表 var oaJobNames = new List() { "商邀主管", "商邀" }; var oaNodeOpUsers = users.Where(u => u.JobName != null && oaJobNames.Contains(u.JobName) ) .Select(u => new UserAndQiWeiUserIdView() { Id = u.Id, Name = u.CnName, QiyeChatUserId = u.QiyeChatUserId }) .ToList(); result.Add(GroupProcessType.Invitation, oaNodeOpUsers); #endregion #region 签证流程 //节点可操作用户列表 var visaNodeOpUsers = users.Where(u => u.JobName != null && u.JobName.Contains("签证") ) .Select(u => new UserAndQiWeiUserIdView() { Id = u.Id, Name = u.CnName, QiyeChatUserId = u.QiyeChatUserId }) .ToList(); result.Add(GroupProcessType.Visa, visaNodeOpUsers); #endregion #region 机票流程 //节点可操作用户列表 var airNodeOpJobNames = new List() { "机票" }; airNodeOpJobNames.AddRange(defaultJobNames); var airNodeOpUsers = users.Where(u => u.JobName != null && airNodeOpJobNames.Contains(u.JobName) ) .Select(u => new UserAndQiWeiUserIdView() { Id = u.Id, Name = u.CnName, QiyeChatUserId = u.QiyeChatUserId }) .ToList(); result.Add(GroupProcessType.AirTicket, airNodeOpUsers); #endregion #region 酒店流程 //节点可操作用户列表 var hotelNodeOpJobNames = new List() { "酒店" }; hotelNodeOpJobNames.AddRange(defaultJobNames); var hotelNodeOpUsers = users.Where(u => u.JobName != null && hotelNodeOpJobNames.Contains(u.JobName) ) .Select(u => new UserAndQiWeiUserIdView() { Id = u.Id, Name = u.CnName, QiyeChatUserId = u.QiyeChatUserId }) .ToList(); result.Add(GroupProcessType.Hotel, hotelNodeOpUsers); #endregion #region 地接流程 //节点可操作用户列表 var opNodeOpJobNames = new List() { "OP" }; opNodeOpJobNames.AddRange(defaultJobNames); var opNodeOpUsers = users.Where(u => u.JobName != null && opNodeOpJobNames.Contains(u.JobName) ) .Select(u => new UserAndQiWeiUserIdView() { Id = u.Id, Name = u.CnName, QiyeChatUserId = u.QiyeChatUserId }) .ToList(); result.Add(GroupProcessType.LocalGuide, opNodeOpUsers); #endregion #region 费用结算流程 //节点可操作用户列表 var feeNodeOpUsers = users.Where(u => u.JobName != null && u.JobName.Contains("会计") && u.CnName.Equals("曾艳") ) .Select(u => new UserAndQiWeiUserIdView() { Id = u.Id, Name = u.CnName, QiyeChatUserId = u.QiyeChatUserId }) .ToList(); result.Add(GroupProcessType.FeeSettle, feeNodeOpUsers); #endregion return result; } public class NodeOpUserInfo { public int Id { get; set; } public string CnName { get; set; } public int CompanyId { get; set; } public string CompanyName { get; set; } public int DepId { get; set; } public string DepName { get; set; } public int JobPostId { get; set; } public string JobName { get; set; } public string QiyeChatUserId { get; set; } } public class GroupProcFullNodeDetails { public int GroupId { get; set; } public string GroupName { get; set; } public int GroupType { get; set; } /// /// 流程类型 /// public GroupProcessType ProcType { get; set; } /// /// 流程顺序 /// public int ProcOrder { get { return (int)ProcType; } } public int NodeOrder { get; set; } public string NodeName { get { return ProcType switch { GroupProcessType.Invitation => NodeOrder switch { 1 => "初期报批文字材料", 2 => "第一轮对接", 3 => "第二轮对接", 4 => "取得邀请函", 5 => "公务等事项确认", 6 => "公务邀请数据有效录入", 7 => "文件上传", _ => "" }, GroupProcessType.Visa => NodeOrder switch { 1 => "签证信息", 2 => "票据上传(明细表、费用票据、保单及超支费用账单)", _ => "" }, GroupProcessType.AirTicket => NodeOrder switch { 1 => "初步拟定航程方案及价格", 2 => "机票占位、续位", 3 => "完成机票采购确认", 4 => "进行出票操作并核查信息", 5 => "机票已出", 6 => "完成机票选座", 7 => "票据上传(机票超支费用账单)", 8 => "票据上传", _ => "" }, GroupProcessType.Hotel => NodeOrder switch { 1 => "按照预算,询价、比价、谈价", 2 => "获取酒店确认函与入住名单核对", 3 => "预订酒店并录入OA", 4 => "行前再次确认酒店相关情况", 5 => "行程结束后整理酒店发票与结算", _ => "" }, GroupProcessType.LocalGuide => NodeOrder switch { 1 => "根据机票方案出框架行程", 2 => "联系并询价地接相关的供应商", 3 => "提交供应商报价及比价表", 4 => "执行采购流程", 5 => "制定最终行程单及出行手册", 6 => "送机", 7 => "最终版报批行程、票据上传", _ => "" }, GroupProcessType.FeeSettle => NodeOrder switch { 1 => "城市间交通报批金额核定", 2 => "团组城市间交通及国际机票数据分配的合理性核对", 3 => "整理统计相关财务资料给到各单位", 4 => "费用结算完毕", _ => "" }, _ => "" }; } } public string NodeTips { get { string msg = string.Empty; string promptMessage = GetPromptMessage(); switch (ProcType) { case GroupProcessType.Invitation: switch (NodeOrder) { case 1: msg = "“完成基础请示、报批日程初稿”,4个工作日(仍需根据客户意见和联系情况及时修改补充所需其他材料,例如印证文件、成果表等,直至终稿)"; break; case 2: msg = "7个工作日,所有报批机构前部联系,邀请机构一个国家不少于4家进行重点对接(4家机构中,其中3家机构需有效对接,其中1家可为付费机构备选)"; break; case 3: msg = "10个工作日,根据最新情况,联系公务机构1/3取得回应;邀请机构基本明确。"; break; case 4: msg = "正式名单下放后2周内(含非工作日)。如团组前期准备时间已经较长,则按客户要求尽快提供。 加急团组备注特殊情况。"; break; case 5: msg = "团组出发前,5个工作日完成所有公务确认工作。"; break; case 6: break; case 7: msg = "如果需要上传请在团组结束前完成"; break; } break; case GroupProcessType.Visa: switch (NodeOrder) { case 1: break; case 2: msg = "按进度实际签证办理落实情况,团组出发前上传票据。"; break; } break; case GroupProcessType.AirTicket: switch (NodeOrder) { case 1: msg = "建团后打勾确认出团的时候开始24小时内。"; break; case 2: break; case 3: msg = "完成机票采购确认(含预算核对、出票确认等)"; break; case 4: break; case 5: msg = "团组出发前2个工作日"; break; case 6: break; case 7: msg = "机票蓝联打票及上传机票超支费用账单,团组归国后5个工作日内"; break; case 8: msg = "1. 票据上传(机票报销蓝联、行程单及机票说明) \r\n 2. 团组归国后10个工作日内 *按机票报价*0.999折扣出具机票报销蓝联、行程单及机票说明"; break; } break; case GroupProcessType.Hotel: switch (NodeOrder) { case 1: msg = "1. 筛选并按照预算标准,对目标酒店进行询价、比价、谈价 \r\n2. 建团后打勾确认出团的时候开始2个工作日。"; break; case 2: break; case 3: break; case 4: msg = "1.行前再次确认酒店订单、付款状态及入住安排 \r\n 2.团组出发前2个工作日"; break; case 5: Days = 5; msg = "1.行程结束后整理酒店发票(含超支费用发票)与结算 \r\n 2.团组结束后5个工作日内"; break; } break; case GroupProcessType.LocalGuide: switch (NodeOrder) { case 1: msg = "机票行程代码最后一段录入后1个工作日内。"; break; case 2: msg = "1.联系并询价地接、餐厅、用车、景点等供应商 \r\n 2. 团组出行前20个工作日"; break; case 3: msg = "上一步往后3个工作日内"; break; case 4: msg = "上一步往后2个工作日内"; break; case 5: msg = "1.制定最终《行程单》及《出行手册》 \r\n2. 倒推表里开行前会 -3天。"; break; case 6: break; case 7: msg = "团组归国后5个工作日内 *上传最终报批行程,确定城市间交通最终版报价分配;地接账单(清楚标注超时及其他项超支费用)、地接交通费用原始票据、城市间交通明细表;"; break; } break; case GroupProcessType.FeeSettle: switch (NodeOrder) { case 1: msg = "团组报批前"; break; case 2: msg = "团组报批前三公费用表"; break; case 3: msg = "1.整理统计团组超支费用、三公报销资料给到各单位 \r\n 2. 团组归国后12个工作日内"; break; case 4: break; } break; } if (!string.IsNullOrEmpty(promptMessage)) { msg = $"{promptMessage}({msg})"; } return msg; } } /// /// 提示时间 - 基数 /// public DateTime? PromptTimeBase { get; set; } /// /// 提前提醒天数 /// public int Days { get; set; } /// /// 是否是工作日 /// true 工作日 false 自然日 /// public bool IsWorkday { get; set; } = false; /// /// 是否提示(包含条件检查) /// public bool IsPrompt { get { // 基础检查:提示时间不为空 if (!PromptTime.HasValue) return false; // 额外条件 // 提示时间不能是未来时间 if (PromptTime > DateTime.Now) return false; return true; } } /// /// 提示时间 /// public DateTime? PromptTime { get { DateTime? dt = null; if (PromptTimeBase.HasValue) { if (IsWorkday) { dt = AddWeekdays(PromptTimeBase.Value, Days); } else { dt = PromptTimeBase.Value.AddDays(Days); } } return dt; } } /// /// 是否预警(基于时间计算规则) /// 业务规则: /// 1:商邀流程只在第三个节点预警 /// 2:预警时间 ≤ 提示时间(当两者都存在时) /// public bool IsAlert { get { // 基本规则:有预警时间 if (!AlertTime.HasValue) return false; // 业务规则1:商邀流程只在第三个节点预警 if (ProcType == GroupProcessType.Invitation && NodeOrder != 3) { return false; // 商邀流程非第三个节点,不预警 } // 业务规则2:预警时间 ≤ 提示时间(当两者都存在时) if (PromptTime.HasValue && AlertTime.Value > PromptTime.Value) return false; return true; } } /// /// 预警时间(基于提示时间计算) /// 规则: /// 1. Days > 0:基准时间向前推(未来) /// 2. Days < 0:基准时间向后推(过去) /// 3. Days = 0:不计算预警时间 /// 4. |Days| = 1:预警时间等于提示时间 /// 5. |Days| > 1:预警天数 = 提示天数的一半(向下取整) /// public DateTime? AlertTime { get { if (!PromptTimeBase.HasValue || Days == 0) return null; // 获取天数的绝对值用于计算比例 int absDays = Math.Abs(Days); int actualDays; if (absDays == 1) { // |Days| = 1 时,预警时间等于提示时间 actualDays = Days; // 保持原方向(正/负) } else { // |Days| > 1 时,预警天数 = 提示天数的一半 int alertDays = Math.Max(1, absDays / 2); // 保持原方向(正/负),但天数是原天数的一半 actualDays = (Days > 0 ? alertDays : -alertDays); } // 统一计算逻辑 if (IsWorkday) { return AddWeekdays(PromptTimeBase.Value, actualDays); } else { return PromptTimeBase.Value.AddDays(actualDays); } } } /// /// 提示人 /// public List PromptPerson { get; set; } public GroupProcFullNodeDetails() { } public static GroupProcFullNodeDetails Create(int groupId, string groupName, int groupType, GroupProcessType procType, int nodeOrder, DateTime? promptTimeBase, int days, bool isWorkday, List promptPerson) { return new GroupProcFullNodeDetails { GroupId = groupId, GroupType = groupType, GroupName = groupName, ProcType = procType, NodeOrder = nodeOrder, PromptTimeBase = promptTimeBase, Days = days, IsWorkday = isWorkday, PromptPerson = promptPerson }; } private string GetPromptMessage() { if (PromptTime.HasValue) { return $"请于{PromptTime.Value:yyyy年MM月dd日}内完成该项工作"; } return string.Empty; } } public class UserAndQiWeiUserIdView { public int Id { get; set; } public string Name { get; set; } public string QiyeChatUserId { get; set; } } #endregion #region 状态变更 可操作任意节点 进行中<-->已完成 双向变更 /// /// 更新节点状态(支持任意节点状态变更,无需处理当前节点) /// /// /// /// /// public async Task UpdateNodeStatusSimpleAsync(int nodeId, int currUserId, ProcessStatus targetStatus = ProcessStatus.Completed) { try { var result = await _sqlSugar.Ado.UseTranAsync(async () => { // 1. 获取节点和流程信息 var node = await _sqlSugar.Queryable() .FirstAsync(n => n.Id == nodeId && n.IsDel == 0); if (node == null) return new Result { Code = StatusCodes.Status400BadRequest, Msg = "当前节点不存在或已被删除。" }; // 2. 用户权限验证 if (!HasNodeOperationPermission(node, currUserId)) { return new Result { Code = StatusCodes.Status400BadRequest, Msg = "当前用户没有操作此节点的权限." }; } var process = await _sqlSugar.Queryable() .FirstAsync(p => p.Id == node.ProcessId && p.IsDel == 0); if (process == null) return new Result { Code = StatusCodes.Status400BadRequest, Msg = "关联的流程不存在。" }; // 3. 检查是否重复操作 if (node.OverallStatus == targetStatus) { return new Result { Code = StatusCodes.Status400BadRequest, Msg = $"当前节点已为{GetStatusDescription(targetStatus)}状态,无需重复操作。。" }; } // 4. 存储更新前的值 var nodeBefore = CloneNode(node); var processBefore = CloneProcess(process); // 5. 更新节点状态 node.OverallStatus = targetStatus; node.Operator = currUserId; node.OperationTime = DateTime.Now; // 6. 保存节点更新 await _sqlSugar.Updateable(node) .UpdateColumns(n => new { n.OverallStatus, n.Operator, n.OperationTime }) .ExecuteCommandAsync(); // 7. 更新流程状态(基于所有节点的状态计算) await UpdateProcessOverallStatusAsync(process, currUserId); // 8. 记录日志 await LogNodeOpAsync(nodeBefore, node, "Update", currUserId); await LogProcessOpAsync(processBefore, process, "Update", currUserId); return new Result { Code = StatusCodes.Status200OK, Msg = "操作成功。" }; }); return result.IsSuccess ? result.Data : new Result { Code = StatusCodes.Status500InternalServerError, Msg = result.ErrorMessage }; } catch (BusinessException ex) { return new Result { Code = StatusCodes.Status400BadRequest, Msg = ex.Message }; } catch (Exception ex) { return new Result { Code = StatusCodes.Status500InternalServerError, Msg = "系统错误,请稍后重试" }; } } /// /// 更新流程整体状态(根据所有节点状态计算) /// /// /// /// private async Task UpdateProcessOverallStatusAsync(Grp_ProcessOverview process, int currUserId) { // 获取所有节点 var allNodes = await _sqlSugar.Queryable() .Where(n => n.ProcessId == process.Id && n.IsDel == 0) .ToListAsync(); // 统计节点状态 int totalCount = allNodes.Count; int completedCount = allNodes.Count(n => n.OverallStatus == ProcessStatus.Completed); int inProgressCount = allNodes.Count(n => n.OverallStatus == ProcessStatus.InProgress); // 判断流程状态 ProcessStatus newProcessStatus; if (completedCount == totalCount) { // 所有节点都已完成 newProcessStatus = ProcessStatus.Completed; process.EndTime = DateTime.Now; } else if (inProgressCount > 0 || completedCount > 0) { // 有节点进行中或已完成 newProcessStatus = ProcessStatus.InProgress; process.EndTime = null; } else { // 所有节点都未开始 newProcessStatus = ProcessStatus.UnStarted; process.EndTime = null; } // 更新流程状态(只有状态变化时才更新) if (process.OverallStatus != newProcessStatus) { process.OverallStatus = newProcessStatus; process.UpdatedUserId = currUserId; process.UpdatedTime = DateTime.Now; await _sqlSugar.Updateable(process) .UpdateColumns(p => new { p.OverallStatus, p.EndTime, p.UpdatedUserId, p.UpdatedTime }) .ExecuteCommandAsync(); } } /// /// 验证用户是否有操作节点的权限 /// private static bool HasNodeOperationPermission(Grp_ProcessNode node, int userId) { // 如果 OpUserList 为空或 null,表示不限制权限 if (node.OpUserList == null || node.OpUserList.Count == 0) { return true; } // 检查当前用户是否在权限列表中 return node.OpUserList.Contains(userId); } /// /// 克隆节点对象 /// /// /// private static Grp_ProcessNode CloneNode(Grp_ProcessNode node) { return new Grp_ProcessNode() { Id = node.Id, ProcessId = node.ProcessId, NodeName = node.NodeName, NodeOrder = node.NodeOrder, OverallStatus = node.OverallStatus, Operator = node.Operator, OperationTime = node.OperationTime, IsCurrent = node.IsCurrent, }; } /// /// 克隆流程对象 /// /// /// private static Grp_ProcessOverview CloneProcess(Grp_ProcessOverview process) { return new Grp_ProcessOverview() { Id = process.Id, GroupId = process.GroupId, ProcessOrder = process.ProcessOrder, ProcessType = process.ProcessType, OverallStatus = process.OverallStatus, StartTime = process.StartTime, EndTime = process.EndTime, UpdatedUserId = process.UpdatedUserId, UpdatedTime = process.UpdatedTime }; } #endregion #region 状态变更 流程导向 /// /// 更新节点状态 /// public async Task UpdateNodeStatusAsync(int nodeId, int currUserId, ProcessStatus processStatus = ProcessStatus.Completed) { try { // 使用事务确保数据一致性 var result = await _sqlSugar.Ado.UseTranAsync(async () => { // 1. 获取并验证节点 var node = await _sqlSugar.Queryable() .FirstAsync(n => n.Id == nodeId && n.IsDel == 0) ?? throw new BusinessException("当前节点不存在或已被删除。"); // 2. 获取流程信息 var process = await _sqlSugar.Queryable() .FirstAsync(p => p.Id == node.ProcessId && p.IsDel == 0) ?? throw new BusinessException("关联的流程不存在。"); // 3. 验证节点操作 ValidateNodeOperation(node, processStatus); // 4. 存储更新前的值 var nodeBefore = new Grp_ProcessNode() { Id = node.Id, ProcessId = node.ProcessId, NodeName = node.NodeName, NodeOrder = node.NodeOrder, OverallStatus = node.OverallStatus, Operator = node.Operator, OperationTime = node.OperationTime, IsCurrent = node.IsCurrent, }; var processBefore = new Grp_ProcessOverview() { Id = process.Id, GroupId = process.GroupId, ProcessOrder = process.ProcessOrder, ProcessType = process.ProcessType, OverallStatus = process.OverallStatus, StartTime = process.StartTime, EndTime = process.EndTime, UpdatedUserId = process.UpdatedUserId, UpdatedTime = process.UpdatedTime }; // 5. 处理特殊状态流转逻辑 if (node.OverallStatus == ProcessStatus.Completed && processStatus == ProcessStatus.InProgress) { // 从已完成回退到进行中 await HandleRollbackToInProgressAsync(node, process, currUserId); } else { // 正常状态更新 await HandleNormalStatusUpdateAsync(node, process, processStatus, currUserId); } // 6. 记录日志 await LogNodeOpAsync(nodeBefore, node, "Update", currUserId); await LogProcessOpAsync(processBefore, process, "Update", currUserId); return new Result { Code = StatusCodes.Status200OK, Msg = "操作成功。" }; }); return result.IsSuccess ? result.Data : new Result { Code = StatusCodes.Status500InternalServerError, Msg = result.ErrorMessage }; } catch (BusinessException ex) { return new Result { Code = StatusCodes.Status400BadRequest, Msg = ex.Message }; } catch (Exception ex) { return new Result { Code = StatusCodes.Status500InternalServerError, Msg = "系统错误,请稍后重试" }; } } /// /// 处理从已完成回退到进行中的逻辑 /// private async Task HandleRollbackToInProgressAsync(Grp_ProcessNode node, Grp_ProcessOverview process, int currUserId) { // 1. 检查是否可以回退 if (process.OverallStatus == ProcessStatus.Completed && process.ProcessType != GroupProcessType.Invitation) { throw new BusinessException("整个流程已完成,无法回退单个节点状态。"); } // 2. 取消其他当前节点,设置当前节点为当前节点 await ClearOtherCurrentNodesAsync(node.ProcessId, node.Id, currUserId); // 3. 更新节点状态 node.OverallStatus = ProcessStatus.InProgress; node.Operator = currUserId; node.OperationTime = DateTime.Now; node.IsCurrent = true; await _sqlSugar.Updateable(node) .UpdateColumns(n => new { n.OverallStatus, n.Operator, n.OperationTime, n.IsCurrent }) .ExecuteCommandAsync(); // 4. 更新流程状态为进行中(如果已完成的流程回退) if (process.OverallStatus == ProcessStatus.Completed) { process.OverallStatus = ProcessStatus.InProgress; process.EndTime = null; // 清空完成时间 process.UpdatedUserId = currUserId; process.UpdatedTime = DateTime.Now; await _sqlSugar.Updateable(process) .UpdateColumns(p => new { p.OverallStatus, p.EndTime, p.UpdatedUserId, p.UpdatedTime }) .ExecuteCommandAsync(); } } /// /// 处理正常状态更新逻辑 /// private async Task HandleNormalStatusUpdateAsync(Grp_ProcessNode node, Grp_ProcessOverview process, ProcessStatus newStatus, int currUserId) { // 1. 更新节点基础信息 node.OverallStatus = newStatus; node.Operator = currUserId; node.OperationTime = DateTime.Now; // 2. 根据新状态处理节点和流程逻辑 if (newStatus == ProcessStatus.InProgress) { // 设置为进行中:需要成为当前节点 await ClearOtherCurrentNodesAsync(node.ProcessId, node.Id, currUserId); node.IsCurrent = true; // 更新流程状态为进行中 process.OverallStatus = ProcessStatus.InProgress; process.UpdatedUserId = currUserId; process.UpdatedTime = DateTime.Now; } else if (newStatus == ProcessStatus.Completed) { // 设置为已完成:取消当前节点状态 node.IsCurrent = false; // 如果是当前节点完成,需要处理流程流转 if (node.IsCurrent || process.ProcessType == GroupProcessType.Invitation) { await HandleNodeCompletionAsync(node, process, currUserId); } else { // 非当前节点完成,检查流程状态 await CheckAndUpdateProcessStatusAsync(process, currUserId); } } else if (newStatus == ProcessStatus.UnStarted) { // 重置为未开始:取消当前节点状态 node.IsCurrent = false; // 更新流程状态 await CheckAndUpdateProcessStatusAsync(process, currUserId); } // 3. 保存节点更新 await _sqlSugar.Updateable(node) .UpdateColumns(n => new { n.OverallStatus, n.Operator, n.OperationTime, n.IsCurrent }) .ExecuteCommandAsync(); // 4. 保存流程更新 await _sqlSugar.Updateable(process) .UpdateColumns(p => new { p.OverallStatus, p.EndTime, p.UpdatedUserId, p.UpdatedTime }) .ExecuteCommandAsync(); } /// /// 处理节点完成后的流程流转(替代原来的 ProcessCurrentNodeCompletionAsync) /// private async Task HandleNodeCompletionAsync(Grp_ProcessNode currentNode, Grp_ProcessOverview process, int currUserId) { // 1. 获取所有节点 var allNodes = await _sqlSugar.Queryable() .Where(x => x.IsDel == 0 && x.ProcessId == currentNode.ProcessId) .ToListAsync(); // 2. 商邀流程特殊处理 if (process.ProcessType == GroupProcessType.Invitation) { int completedCount = allNodes.Count(n => n.OverallStatus == ProcessStatus.Completed); int nodeCount = allNodes.Count; if (completedCount == nodeCount) // 所有节点完成 { process.OverallStatus = ProcessStatus.Completed; process.EndTime = DateTime.Now; } else if (completedCount > 0) { process.OverallStatus = ProcessStatus.InProgress; } } else { // 3. 普通流程:查找下一个节点 var nextNode = allNodes .Where(n => n.NodeOrder == currentNode.NodeOrder + 1) .FirstOrDefault(); if (nextNode != null) { // 激活下一个节点 var nextNodeBefore = new Grp_ProcessNode() { Id = nextNode.Id, ProcessId = nextNode.ProcessId, NodeName = nextNode.NodeName, NodeOrder = nextNode.NodeOrder, OverallStatus = nextNode.OverallStatus, Operator = nextNode.Operator, OperationTime = nextNode.OperationTime, IsCurrent = nextNode.IsCurrent, }; nextNode.IsCurrent = true; nextNode.OverallStatus = ProcessStatus.InProgress; nextNode.Operator = currUserId; nextNode.OperationTime = DateTime.Now; await _sqlSugar.Updateable(nextNode) .UpdateColumns(n => new { n.IsCurrent, n.OverallStatus, n.Operator, n.OperationTime }) .ExecuteCommandAsync(); await LogNodeOpAsync(nextNodeBefore, nextNode, "Start", currUserId); process.OverallStatus = ProcessStatus.InProgress; } else { // 没有下一个节点,流程完成 process.OverallStatus = ProcessStatus.Completed; process.EndTime = DateTime.Now; } } process.UpdatedUserId = currUserId; process.UpdatedTime = DateTime.Now; } /// /// 清除其他当前节点 /// private async Task ClearOtherCurrentNodesAsync(int processId, int currentNodeId, int currUserId) { var otherCurrentNodes = await _sqlSugar.Queryable() .Where(n => n.ProcessId == processId && n.Id != currentNodeId && n.IsCurrent && n.IsDel == 0) .ToListAsync(); foreach (var otherNode in otherCurrentNodes) { var before = new Grp_ProcessNode() { Id = otherNode.Id, ProcessId = otherNode.ProcessId, NodeName = otherNode.NodeName, NodeOrder = otherNode.NodeOrder, OverallStatus = otherNode.OverallStatus, Operator = otherNode.Operator, OperationTime = otherNode.OperationTime, IsCurrent = otherNode.IsCurrent, }; otherNode.IsCurrent = false; await _sqlSugar.Updateable(otherNode) .UpdateColumns(n => new { n.IsCurrent }) .ExecuteCommandAsync(); await LogNodeOpAsync(before, otherNode, "Update", currUserId); } } /// /// 检查并更新流程状态 /// private async Task CheckAndUpdateProcessStatusAsync(Grp_ProcessOverview process, int currUserId) { // 获取所有节点状态 var allNodes = await _sqlSugar.Queryable() .Where(x => x.IsDel == 0 && x.ProcessId == process.Id) .ToListAsync(); int totalCount = allNodes.Count; int completedCount = allNodes.Count(n => n.OverallStatus == ProcessStatus.Completed); int inProgressCount = allNodes.Count(n => n.OverallStatus == ProcessStatus.InProgress); // 更新流程状态 if (completedCount == totalCount) { process.OverallStatus = ProcessStatus.Completed; process.EndTime = DateTime.Now; } else if (completedCount > 0 || inProgressCount > 0) { process.OverallStatus = ProcessStatus.InProgress; process.EndTime = null; } else { process.OverallStatus = ProcessStatus.UnStarted; process.EndTime = null; } process.UpdatedUserId = currUserId; process.UpdatedTime = DateTime.Now; } /// /// 验证节点操作权限 /// private static void ValidateNodeOperation(Grp_ProcessNode node, ProcessStatus targetStatus) { // 验证状态流转是否合法 if (node.OverallStatus == targetStatus) { throw new BusinessException($"当前节点已为{GetStatusDescription(targetStatus)}状态,无需重复操作。"); } // 允许的流转规则: // 1. 未开始 → 进行中 → 已完成(正常流程) // 2. 已完成 → 进行中(回退操作) // 3. 进行中 → 未开始(重置操作) // 禁止的操作:已完成 → 未开始(需要先回到进行中) if (node.OverallStatus == ProcessStatus.Completed && targetStatus == ProcessStatus.UnStarted) { throw new BusinessException("已完成节点不可直接更改为未开始状态,请先更改为进行中状态。"); } } private static string GetStatusDescription(ProcessStatus status) { return status switch { ProcessStatus.UnStarted => "未开始", ProcessStatus.InProgress => "进行中", ProcessStatus.Completed => "已完成", _ => "未知状态" }; } #endregion /// /// 更新签证节点信息及状态 /// /// 签证节点更新数据传输对象 /// 操作结果 public async Task UpdateVisaNodeDetailsAsync(GroupProcessUpdateVisaNodeDetailsDto dto) { // 1. 获取并验证节点和流程 var node = await _sqlSugar.Queryable() .FirstAsync(n => n.Id == dto.NodeId && n.IsDel == 0) ?? throw new BusinessException("当前节点不存在或已被删除。"); var process = await _sqlSugar.Queryable() .FirstAsync(p => p.Id == node.ProcessId && p.IsDel == 0) ?? throw new BusinessException("当前流程不存在或已被删除。"); if (process.ProcessType != GroupProcessType.Visa) { throw new BusinessException("当前流程节点不为签证流程,不可编辑。"); } // 2. 检查签证子节点 字段信息是否全部填写 var allSubNodesCompleted = dto.VisaSubNodes?.All(subNode => EntityExtensions.IsCompleted(subNode)) ?? false; // 2.1 存储更新前流程及节点信息 var nodeBefore = new Grp_ProcessNode() { Id = node.Id, ProcessId = node.ProcessId, NodeName = node.NodeName, NodeOrder = node.NodeOrder, OverallStatus = node.OverallStatus, Operator = node.Operator, OperationTime = node.OperationTime, IsCurrent = node.IsCurrent, }; var processBefore = new Grp_ProcessOverview() { Id = process.Id, GroupId = process.GroupId, ProcessOrder = process.ProcessOrder, ProcessType = process.ProcessType, OverallStatus = process.OverallStatus, StartTime = process.StartTime, EndTime = process.EndTime, UpdatedUserId = process.UpdatedUserId, UpdatedTime = process.UpdatedTime }; // 3. 更新节点信息 node.Remark = JsonConvert.SerializeObject(dto.VisaSubNodes); node.Operator = dto.CurrUserId; node.OperationTime = DateTime.Now; if (allSubNodesCompleted) { node.OverallStatus = ProcessStatus.Completed; process.OverallStatus = ProcessStatus.Completed; process.EndTime = DateTime.Now; process.UpdatedUserId = dto.CurrUserId; process.UpdatedTime = DateTime.Now; // 更新流程状态 await _sqlSugar.Updateable(process) .UpdateColumns(p => new { p.OverallStatus, p.EndTime, p.UpdatedUserId, p.UpdatedTime }) .ExecuteCommandAsync(); //记录流程日志 await LogProcessOpAsync(processBefore, process, "Update", dto.CurrUserId); } // 4. 保存节点更新 await _sqlSugar.Updateable(node) .UpdateColumns(n => new { n.Remark, n.Operator, n.OperationTime, n.OverallStatus }) .ExecuteCommandAsync(); //记录节点日志 await LogNodeOpAsync(nodeBefore, node, "Update", dto.CurrUserId); return new Result { Code = 200, Msg = "节点信息更新成功。" }; } /// /// 更新节点信息及状态 /// /// 更新节点信息及状态 /// 操作结果 public async Task SetActualDoneAsync(GroupProcessSetActualDoneDto dto) { int nodeId = dto.NodeId; var isDtNul = DateTime.TryParse(dto.ActualDone, out DateTime actualDone); int currUserId = dto.CurrUserId; bool isAssist = dto.IsAssist; bool isFileUp = dto.IsFileUp; bool isPart = dto.IsPart; // 1. 获取并验证节点和流程 var node = await _sqlSugar.Queryable() .FirstAsync(n => n.Id == nodeId && n.IsDel == 0); if (node == null) return new Result { Code = 400, Msg = "当前节点不存在或已被删除。" }; // 1.1. 用户权限验证 if (!HasNodeOperationPermission(node, currUserId)) { return new Result { Code = 400, Msg = "当前用户没有操作此节点的权限。" }; } var process = await _sqlSugar.Queryable() .FirstAsync(p => p.Id == node.ProcessId && p.IsDel == 0); if (process == null) return new Result { Code = 400, Msg = "当前流程不存在或已被删除。" }; // 2.1 存储更新前流程及节点信息 var nodeBefore = CloneNode(node); if (isDtNul) { node.ActualDone = actualDone; } else node.ActualDone = null; node.IsAssist = isAssist; node.IsFileUp = isFileUp; node.IsPart = isPart; node.Remark = dto.Remark; // 3. 保存节点更新 await _sqlSugar.Updateable(node) .UpdateColumns(n => new { n.ActualDone, n.IsAssist, n.IsFileUp, n.IsPart, n.Remark, }) .ExecuteCommandAsync(); //记录节点日志 await LogNodeOpAsync(nodeBefore, node, "Update", currUserId); //当前节点未完成则更改节点状态为已完成 if (node.OverallStatus == ProcessStatus.InProgress && isDtNul) { var statusResult = await UpdateNodeStatusSimpleAsync(node.Id,dto.CurrUserId); //日志记录执行结果 _logger.LogInformation($"团组流程设置完成时间:调用更改状态接口(SetActualDoneAsync -> UpdateNodeStatusAsync):[状态变更:进行中 -> 已完成] result: Code={statusResult.Code}, Msg={statusResult.Msg}"); } //当前节点 实际完成时间设置为空且当前状态为已完成 则更改当前节点状态为进行中 else if (node.OverallStatus == ProcessStatus.Completed && !isDtNul) { var statusResult = await UpdateNodeStatusSimpleAsync(node.Id, dto.CurrUserId, ProcessStatus.InProgress); //日志记录执行结果 _logger.LogInformation($"团组流程设置完成时间调:用更改状态接口(SetActualDoneAsync -> UpdateNodeStatusAsync):[状态变更:已完成 -> 进行中] result: Code={statusResult.Code}, Msg={statusResult.Msg}"); } return new Result { Code = 200, Msg = "设置成功。" }; } #endregion #region 会务流程 /// /// 设置节点流程模板 /// /// /// /// public async Task> DefaultConfProcessTemps(int groupId, int currUserId) { var temps = new List(); //团组验证 var groupInfo = await _sqlSugar.Queryable().FirstAsync(g => g.Id == groupId); if (groupInfo == null) return temps; //// 检查是否已存在流程 //var existingProcesses = await _sqlSugar.Queryable() // .Where(p => p.IsDel == 0 && p.GroupId == groupId) // .ToListAsync(); //if (existingProcesses.Any()) return temps; var users = await _sqlSugar.Queryable() .LeftJoin((u, c) => u.CompanyId == c.Id) .LeftJoin((u, c, d) => u.DepId == d.Id) .LeftJoin((u, c, d, jp) => u.JobPostId == jp.Id) .Where((u, c, d, jp) => u.IsDel == 0) .Select((u, c, d, jp) => new { u.Id, u.CnName, u.CompanyId, c.CompanyName, u.DepId, d.DepName, u.JobPostId, jp.JobName, }) .ToListAsync(); //节点可操作用户列表 var nodeOpUsers = users.Where(u => u.JobName != null && u.DepName.Contains("策划部") ).Select(u => u.Id) .ToList(); #region 会务流程 //参与人 participators var defaultParticipators = new List() { 213, //李新江 }; var defaultPorc1 = new List() { Grp_ConfProcessNode.Create(1,"方案/报价(含成本)","", ProcessStatus.InProgress,true,false, currUserId,defaultParticipators,nodeOpUsers), Grp_ConfProcessNode.Create(2,"项目前期比选/招投标相关文件","", ProcessStatus.InProgress,false,false, currUserId,defaultParticipators,nodeOpUsers), Grp_ConfProcessNode.Create(3,"参与投标","", ProcessStatus.InProgress,false,false, currUserId,defaultParticipators,nodeOpUsers), Grp_ConfProcessNode.Create(4,"拟定/签订合同","", ProcessStatus.InProgress,false,false, currUserId,defaultParticipators,nodeOpUsers), Grp_ConfProcessNode.Create(5,"场地预订/物料设计/对接活动所需的供应商/嘉宾邀约","", ProcessStatus.InProgress,false,false, currUserId,defaultParticipators,nodeOpUsers), Grp_ConfProcessNode.Create(6,"现场执行","", ProcessStatus.InProgress,false,false, currUserId,defaultParticipators,nodeOpUsers), Grp_ConfProcessNode.Create(7,"验收报告/决算表","", ProcessStatus.InProgress,false,false, currUserId,defaultParticipators,nodeOpUsers), Grp_ConfProcessNode.Create(8,"跟进项目收款","", ProcessStatus.InProgress,false,true, currUserId,defaultParticipators,nodeOpUsers), //Grp_ConfProcessNode.Create(9,"票据上传","该项目相关票据", ProcessStatus.InProgress,false,true, currUserId,defaultParticipators,nodeOpUsers), }; temps.Add(Grp_ConfProcessOverview.Create(groupId, 1, ProcessStatus.InProgress, currUserId, defaultPorc1)); var defaultPorc2 = new List() { Grp_ConfProcessNode.Create(1,"方案/报价(含成本)","", ProcessStatus.InProgress,true,false, currUserId,defaultParticipators,nodeOpUsers), Grp_ConfProcessNode.Create(2,"拟定/签订合同","", ProcessStatus.InProgress,false,false, currUserId,defaultParticipators,nodeOpUsers), Grp_ConfProcessNode.Create(3,"场地预订/物料设计/对接活动所需的供应商/嘉宾邀约","", ProcessStatus.InProgress,false,false, currUserId,defaultParticipators,nodeOpUsers), Grp_ConfProcessNode.Create(4,"现场执行","", ProcessStatus.InProgress,false,false, currUserId,defaultParticipators,nodeOpUsers), Grp_ConfProcessNode.Create(5,"验收报告/决算表","", ProcessStatus.InProgress,false,false, currUserId,defaultParticipators,nodeOpUsers), Grp_ConfProcessNode.Create(6,"跟进项目收款","", ProcessStatus.InProgress,false,true, currUserId,defaultParticipators,nodeOpUsers), //Grp_ConfProcessNode.Create(7,"票据上传","该项目相关票据", ProcessStatus.InProgress,false,true, currUserId,defaultParticipators,nodeOpUsers), }; temps.Add(Grp_ConfProcessOverview.Create(groupId, 2, ProcessStatus.InProgress, currUserId, defaultPorc2)); #endregion return temps; } /// /// 团组会务流程初始化 /// /// 团组Id /// 当前用户Id /// 节点模板Id /// 创建的流程信息 public async Task ConfProcessInitAsync(int groupId, int currUserId,int nodeTempId = 1) { //团组验证 var groupInfo = await _sqlSugar.Queryable().FirstAsync(g => g.Id == groupId); if (groupInfo == null) { return new Result { Code = 400, Msg = "团组不存在" }; } // 检查是否已存在流程 var existingProcesses = await _sqlSugar.Queryable() .Where(p => p.IsDel == 0 && p.GroupId == groupId) .ToListAsync(); if (existingProcesses.Any()) { return new Result { Code = 400, Msg = "团组会务流程已存在" }; } // 定义默认的流程节点 var temps = await DefaultConfProcessTemps(groupId, currUserId); if (temps == null || temps.Count < 1) return new Result { Code = 400, Msg = "团组会务默认流程不存在" }; var process = temps.FirstOrDefault(x => x.ProcessOrder == nodeTempId); process.CreateUserId = currUserId; _sqlSugar.BeginTran(); try { var processId = await _sqlSugar.Insertable(process).ExecuteReturnIdentityAsync(); if (processId < 1) { _sqlSugar.RollbackTran(); return new Result { Code = 400, Msg = "团组会务流程进度总览添加失败!" }; } process.Id = processId; // 记录流程日志 await LogConfProcessOpAsync(null, process, "Create", currUserId); var nodes = process.Nodes.Select((nodeDto, index) => new Grp_ConfProcessNode { ProcessId = processId, NodeName = nodeDto.NodeName, NodeOrder = nodeDto.NodeOrder, Participators = nodeDto.Participators, OpUserList = nodeDto.OpUserList, OverallStatus = nodeDto.OverallStatus, NodeDescTips = nodeDto.NodeDescTips, //Country = nodeDto.Country, IsCurrent = nodeDto.IsCurrent, IsFileUp = nodeDto.IsFileUp, Remark = nodeDto.Remark, CreateUserId = currUserId, }).ToList(); var nodeIds = await _sqlSugar.Insertable(nodes).ExecuteCommandAsync(); if (nodeIds < 1) { _sqlSugar.RollbackTran(); return new Result { Code = 400, Msg = "团组流程进度流程节点添加失败!" }; } //设置节点ID nodes = await _sqlSugar.Queryable().Where(x => x.IsDel == 0 && x.ProcessId == processId).ToListAsync(); //记录节点日志 foreach (var node in nodes) { await LogConfNodeOpAsync(null, node, "Create", currUserId); } _sqlSugar.CommitTran(); return new Result { Code = 200, Msg = "添加成功!" }; } catch (Exception ex) { _sqlSugar.RollbackTran(); return new Result { Code = 500, Msg = $"操作失败!msg:{ex.Message}" }; } } /// /// 获取团组会务流程及节点详情 /// /// 团组Id /// 当前用户Id /// public async Task ConfProcessesDetailsAsync(int groupId, int currUserId = 4) { //团组验证 var groupInfo = await _sqlSugar.Queryable().FirstAsync(g => g.Id == groupId); if (groupInfo == null) { return new Result { Code = 400, Msg = "团组不存在" }; } // 检查是否已存在流程 var existingProcesses = await _sqlSugar.Queryable() .Where(p => p.IsDel == 0 && p.GroupId == groupId) .ToListAsync(); if (!existingProcesses.Any()) { //新建团组流程 var res = await ConfProcessInitAsync(groupId, currUserId); if (res.Code != 200) { return res; } } var users = await _sqlSugar.Queryable().Select(x => new { x.Id, x.CnName }).ToListAsync(); var processData = await _sqlSugar.Queryable() .Where(p => p.GroupId == groupId && p.IsDel == 0) .Mapper(p => p.Nodes, p => p.Nodes.First().ProcessId) .ToListAsync(); // 预先构建用户字典,提升查询性能 var userDict = users.ToDictionary(u => u.Id, u => u.CnName); bool isNodeTemplSwitchable = true; var processes = processData.Select(p => { var orderedNodes = p.Nodes.OrderBy(n => n.NodeOrder).ToList(); var totalNodes = orderedNodes.Count; isNodeTemplSwitchable = p.OverallStatus != ProcessStatus.Completed; return new ConfProcessOverInfoView() { Id = p.Id, GroupId = p.GroupId, ProcessType = p.ProcessType, ProcessName = p.ProcessType.GetEnumDescription(), Nodes = orderedNodes.Select((n, index) => { var isLastNode = index == totalNodes - 1; // 文件上传按钮启用规则 bool isEnaFileUpBtn = false; if (isLastNode) { isEnaFileUpBtn = true; } // 获取操作人姓名(使用字典提升性能) string operatorName = "-"; if (n.Operator.HasValue && userDict.TryGetValue(n.Operator.Value, out var name)) { operatorName = name; } return new ConfProcessNodeInfoView() { Id = n.Id, ProcessId = n.ProcessId, NodeOrder = n.NodeOrder, NodeName = n.NodeName, NodeDescTips = n.NodeDescTips, OverallStatus = n.OverallStatus, Participators = n.Participators, OpUserList = n.OpUserList, StatusText = n.OverallStatus.GetEnumDescription(), Operator = operatorName, OpeateTime = n.OperationTime?.ToString("yyyy-MM-dd HH:mm:ss") ?? "-", ActualDone = n.ActualDone?.ToString("yyyy-MM-dd HH:mm:ss") ?? "", IsEnaFileUpBtn = isEnaFileUpBtn, IsFileUp = n.IsFileUp, // 票据上传节点 存储值 Remark = n.Remark }; }).ToList() }; }).ToList(); var view = new ConfProcessOverInfo() { IsNodeTemplSwitchable = isNodeTemplSwitchable, ConfProcess = processes }; return new Result { Code = 200, Data = view, Msg = "查询成功!" }; } /// /// 更新节点模板 /// /// 团组Id /// 节点模板Id /// 当前用户Id /// public async Task ConfProcessChangeNodeTempSaveAsync(int groupId, int nodeTempId, int currUserId) { //节点模板id验证 var nodeTempIds = new List() { 1, 2 }; if (!nodeTempIds.Contains(nodeTempId)) { return new Result { Code = 400, Msg = "请传入有效的节点模板Id" }; } //团组验证 var groupInfo = await _sqlSugar.Queryable().FirstAsync(g => g.Id == groupId); if (groupInfo == null) { return new Result { Code = 400, Msg = "团组不存在" }; } _sqlSugar.BeginTran(); try { // 检查是否已存在流程 var existingProcesses = await _sqlSugar.Queryable() .Where(p => p.IsDel == 0 && p.GroupId == groupId) .ToListAsync(); if (existingProcesses.Any()) { // 用户权限验证 var firstNode = await _sqlSugar.Queryable().FirstAsync(x => x.IsDel == 0 && x.ProcessId == existingProcesses.FirstOrDefault().Id); if (!HasConfNodeOperationPermission(firstNode, currUserId)) { return new Result { Code = 400, Msg = "当前用户没有操作此节点的权限。" }; } //团组会流程完成 不可切换模板 if (existingProcesses.Where(x => x.OverallStatus == ProcessStatus.Completed).ToList().Count > 0) { _sqlSugar.RollbackTran(); return new Result { Code = 400, Msg = $"当前团组会务流程已完成,不可切换节点模板。" }; } //删除 原有的节点模板 var parentIds = existingProcesses.Select(x => x.Id).ToList(); var updProcesses = await _sqlSugar.Updateable() .SetColumns(x => x.DeleteUserId == currUserId) .SetColumns(x => x.DeleteTime == DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")) .SetColumns(x => x.IsDel == 1) .Where(x => parentIds.Contains(x.Id)) .ExecuteCommandAsync(); var updProcessNodes = await _sqlSugar.Updateable() .SetColumns(x => x.DeleteUserId == currUserId) .SetColumns(x => x.DeleteTime == DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")) .SetColumns(x => x.IsDel == 1) .Where(x => parentIds.Contains(x.ProcessId)) .ExecuteCommandAsync(); } //更新模板节点信息 var result = await ConfProcessInitAsync(groupId,currUserId,nodeTempId); if (result.Code == 200) { var infoResult = await ConfProcessesDetailsAsync(groupId, currUserId); if (infoResult.Code == 200) { _sqlSugar.CommitTran(); return infoResult; } } _sqlSugar.RollbackTran(); return new Result { Code = 500, Msg = $"操作失败!Msg:{result.Msg}" }; } catch (Exception ex) { _sqlSugar.RollbackTran(); return new Result { Code = 500, Msg = $"操作失败!Msg:{ex.Message}" }; } } #region 状态变更 可操作任意节点 进行中<-->已完成 双向变更 /// /// 更新节点状态(支持任意节点状态变更,无需处理当前节点) /// /// /// /// /// public async Task UpdateConfNodeStatusSimpleAsync(int nodeId, int currUserId, ProcessStatus targetStatus = ProcessStatus.Completed) { try { var result = await _sqlSugar.Ado.UseTranAsync(async () => { // 1. 获取节点和流程信息 var node = await _sqlSugar.Queryable() .FirstAsync(n => n.Id == nodeId && n.IsDel == 0); if (node == null) return new Result { Code = StatusCodes.Status400BadRequest, Msg = "当前节点不存在或已被删除。" }; // 2. 用户权限验证 if (!HasConfNodeOperationPermission(node, currUserId)) { return new Result { Code = StatusCodes.Status400BadRequest, Msg = "当前用户没有操作此节点的权限。" }; } var process = await _sqlSugar.Queryable() .FirstAsync(p => p.Id == node.ProcessId && p.IsDel == 0); if (process == null) return new Result { Code = StatusCodes.Status400BadRequest, Msg = "关联的流程不存在。" }; // 3. 检查是否重复操作 if (node.OverallStatus == targetStatus) { return new Result { Code = StatusCodes.Status400BadRequest, Msg = "当前节点已为{GetStatusDescription(targetStatus)}状态,无需重复操作。。" }; } // 4. 存储更新前的值 var nodeBefore = CloneConfNode(node); var processBefore = CloneConfProcess(process); // 5. 更新节点状态 node.OverallStatus = targetStatus; node.Operator = currUserId; node.OperationTime = DateTime.Now; // 6. 保存节点更新 await _sqlSugar.Updateable(node) .UpdateColumns(n => new { n.OverallStatus, n.Operator, n.OperationTime }) .ExecuteCommandAsync(); // 7. 更新流程状态(基于所有节点的状态计算) await UpdateConfProcessOverallStatusAsync(process, currUserId); // 8. 记录日志 await LogConfNodeOpAsync(nodeBefore, node, "Update", currUserId); await LogConfProcessOpAsync(processBefore, process, "Update", currUserId); return new Result { Code = StatusCodes.Status200OK, Msg = "操作成功。" }; }); return result.IsSuccess ? result.Data : new Result { Code = StatusCodes.Status500InternalServerError, Msg = result.ErrorMessage }; } catch (BusinessException ex) { return new Result { Code = StatusCodes.Status400BadRequest, Msg = ex.Message }; } catch (Exception ex) { return new Result { Code = StatusCodes.Status500InternalServerError, Msg = "系统错误,请稍后重试" }; } } /// /// 更新流程整体状态(根据所有节点状态计算) /// /// /// /// private async Task UpdateConfProcessOverallStatusAsync(Grp_ConfProcessOverview process, int currUserId) { // 获取所有节点 var allNodes = await _sqlSugar.Queryable() .Where(n => n.ProcessId == process.Id && n.IsDel == 0) .ToListAsync(); // 统计节点状态 int totalCount = allNodes.Count; int completedCount = allNodes.Count(n => n.OverallStatus == ProcessStatus.Completed); int inProgressCount = allNodes.Count(n => n.OverallStatus == ProcessStatus.InProgress); // 判断流程状态 ProcessStatus newProcessStatus; if (completedCount == totalCount) { // 所有节点都已完成 newProcessStatus = ProcessStatus.Completed; process.EndTime = DateTime.Now; } else if (inProgressCount > 0 || completedCount > 0) { // 有节点进行中或已完成 newProcessStatus = ProcessStatus.InProgress; process.EndTime = null; } else { // 所有节点都未开始 newProcessStatus = ProcessStatus.UnStarted; process.EndTime = null; } // 更新流程状态(只有状态变化时才更新) if (process.OverallStatus != newProcessStatus) { process.OverallStatus = newProcessStatus; process.UpdatedUserId = currUserId; process.UpdatedTime = DateTime.Now; await _sqlSugar.Updateable(process) .UpdateColumns(p => new { p.OverallStatus, p.EndTime, p.UpdatedUserId, p.UpdatedTime }) .ExecuteCommandAsync(); } } /// /// 验证用户是否有操作节点的权限 /// private static bool HasConfNodeOperationPermission(Grp_ConfProcessNode node, int userId) { // 如果 OpUserList 为空或 null,表示不限制权限 if (node.OpUserList == null || node.OpUserList.Count == 0) { return true; } // 检查当前用户是否在权限列表中 return node.OpUserList.Contains(userId); } /// /// 克隆节点对象 /// /// /// private static Grp_ConfProcessNode CloneConfNode(Grp_ConfProcessNode node) { return new Grp_ConfProcessNode() { Id = node.Id, ProcessId = node.ProcessId, NodeName = node.NodeName, NodeDescTips = node.NodeDescTips, NodeOrder = node.NodeOrder, OverallStatus = node.OverallStatus, Participators = node.Participators, Operator = node.Operator, OperationTime = node.OperationTime, IsCurrent = node.IsCurrent, ActualDone = node.ActualDone, IsFileUp = node.IsFileUp, OpUserList = node.OpUserList, }; } /// /// 克隆流程对象 /// /// /// private static Grp_ConfProcessOverview CloneConfProcess(Grp_ConfProcessOverview process) { return new Grp_ConfProcessOverview() { Id = process.Id, GroupId = process.GroupId, ProcessOrder = process.ProcessOrder, ProcessType = process.ProcessType, OverallStatus = process.OverallStatus, StartTime = process.StartTime, EndTime = process.EndTime, UpdatedUserId = process.UpdatedUserId, UpdatedTime = process.UpdatedTime }; } #endregion #region 更新节点状态 流程导向 /// /// 更新团组会务流程节点状态 /// /// 节点ID /// 当前用户ID /// 流程状态,默认为已完成 /// 操作结果 public async Task UpdateConfNodeStatusAsync(int nodeId, int currUserId, ProcessStatus processStatus = ProcessStatus.Completed) { try { // 使用事务确保数据一致性 var result = await _sqlSugar.Ado.UseTranAsync(async () => { // 1. 获取并验证节点 var node = await _sqlSugar.Queryable() .FirstAsync(n => n.Id == nodeId && n.IsDel == 0) ?? throw new BusinessException("当前节点不存在或已被删除。"); // 2. 获取流程信息,检查ProcessType var process = await _sqlSugar.Queryable() .FirstAsync(p => p.Id == node.ProcessId && p.IsDel == 0) ?? throw new BusinessException("关联的流程不存在。"); // 3. 节点操作验证 ValidateConfNodeOperation(node, processStatus); // 4. 存储更新前的值 var before = CloneConfNode(node); // 5. 更新节点状态 node.OverallStatus = processStatus; node.Operator = currUserId; node.OperationTime = DateTime.Now; var updateCount = await _sqlSugar.Updateable(node) .UpdateColumns(n => new { n.OverallStatus, n.Operator, n.OperationTime }) .ExecuteCommandAsync(); if (updateCount == 0) { throw new BusinessException("节点状态更新失败。"); } // 6. 记录节点日志 await LogConfNodeOpAsync(before, node, "Update", currUserId); // 7. 如果是完成当前节点,处理流程流转 if (processStatus == ProcessStatus.Completed) { await ConfProcessCurrentNodeCompletionAsync(node, currUserId); } return new Result { Code = StatusCodes.Status200OK, Msg = "操作成功。" }; }); return result.IsSuccess ? result.Data : new Result { Code = StatusCodes.Status500InternalServerError, Msg = result.ErrorMessage }; } catch (BusinessException ex) { // 业务异常 return new Result { Code = StatusCodes.Status400BadRequest, Msg = ex.Message }; } catch (Exception ex) { // 系统异常 return new Result { Code = StatusCodes.Status500InternalServerError, Msg = "系统错误,请稍后重试" }; } } /// /// 验证会务流程节点操作权限 /// /// 流程节点 /// 目标状态 private static void ValidateConfNodeOperation(Grp_ConfProcessNode node, ProcessStatus targetStatus) { // 验证节点是否已完成 if (node.OverallStatus == ProcessStatus.Completed) { throw new BusinessException("当前节点已完成,不可重复操作。"); } // 验证状态流转是否合法(可选) //if (targetStatus != ProcessStatus.Completed) //{ // throw new BusinessException("未开始或者进行中的节点只能重新完成,不可进行其他操作。"); //} // 验证是否尝试将已完成节点改为其他状态 if (node.OverallStatus == ProcessStatus.Completed && targetStatus != ProcessStatus.Completed) { throw new BusinessException("已完成节点不可修改状态。"); } } /// /// 处理当前会务流程节点完成后的流程流转 /// private async Task ConfProcessCurrentNodeCompletionAsync(Grp_ConfProcessNode currentNode, int currUserId) { // 1. 获取流程信息 var process = await _sqlSugar.Queryable() .FirstAsync(p => p.Id == currentNode.ProcessId && p.IsDel == 0) ?? throw new BusinessException("关联的流程不存在。"); var processBefore = new Grp_ConfProcessOverview() { Id = process.Id, GroupId = process.GroupId, ProcessOrder = process.ProcessOrder, ProcessType = process.ProcessType, OverallStatus = process.OverallStatus, StartTime = process.StartTime, EndTime = process.EndTime, UpdatedUserId = process.UpdatedUserId, UpdatedTime = process.UpdatedTime }; // 2. 取消当前节点的当前状态 var before = new Grp_ConfProcessNode() { Id = currentNode.Id, ProcessId = currentNode.ProcessId, NodeName = currentNode.NodeName, NodeOrder = currentNode.NodeOrder, OverallStatus = currentNode.OverallStatus, Operator = currentNode.Operator, OperationTime = currentNode.OperationTime, IsCurrent = currentNode.IsCurrent, }; currentNode.IsCurrent = false; await _sqlSugar.Updateable(currentNode) .UpdateColumns(n => new { n.IsCurrent }) .ExecuteCommandAsync(); // 2.1 记录节点日志 取消当前节点状态 await LogConfNodeOpAsync(before, currentNode, "Update", currUserId); // 3. 查找并激活下一个节点 var nextNode = await _sqlSugar.Queryable() .Where(n => n.ProcessId == currentNode.ProcessId && n.NodeOrder == currentNode.NodeOrder + 1 && n.IsDel == 0) .FirstAsync(); if (nextNode != null) { var nextNodeBefore = new Grp_ConfProcessNode() { Id = nextNode.Id, ProcessId = nextNode.ProcessId, NodeName = nextNode.NodeName, NodeOrder = nextNode.NodeOrder, OverallStatus = nextNode.OverallStatus, Operator = nextNode.Operator, OperationTime = nextNode.OperationTime, IsCurrent = nextNode.IsCurrent, }; // 激活下一个节点 nextNode.IsCurrent = true; nextNode.OverallStatus = ProcessStatus.InProgress; var updateCount = await _sqlSugar.Updateable(nextNode) .UpdateColumns(n => new { n.IsCurrent, n.OverallStatus, n.Operator, n.OperationTime }) .ExecuteCommandAsync(); if (updateCount == 0) { throw new BusinessException("激活下一节点失败"); } // 1.1 记录节点日志 激活下一节点当前节点状态 await LogConfNodeOpAsync(nextNodeBefore, nextNode, "Start", currUserId); // 更新流程状态为进行中 process.OverallStatus = ProcessStatus.InProgress; } else { // 下一节点不存在,整个流程完成 process.OverallStatus = ProcessStatus.Completed; process.EndTime = DateTime.Now; } // 4. 更新流程信息 process.UpdatedUserId = currUserId; process.UpdatedTime = DateTime.Now; var processUpdateCount = await _sqlSugar.Updateable(process) .UpdateColumns(p => new { p.OverallStatus, p.EndTime, p.UpdatedUserId, p.UpdatedTime }) .ExecuteCommandAsync(); if (processUpdateCount == 0) { throw new BusinessException("流程状态更新失败。"); } //记录流程日志 await LogConfProcessOpAsync(processBefore, process, "Update", currUserId); } #endregion /// /// 更新节点信息 /// /// 更新节点信息 /// 操作结果 public async Task SetNodeInfoAsync(ConfProcessSetActualDoneDto dto) { //参与人验证 if (dto.Participators?.Any() != true) return new Result { Code = 400, Msg = "参与人不能为空。" }; var isDtNul = DateTime.TryParse(dto.ActualDone, out DateTime actualDone); //if (!isDtNul) throw new BusinessException("实际操作时间为空或者格式不对。"); int nodeId = dto.NodeId; int currUserId = dto.CurrUserId; bool isFileUp = dto.IsFileUp; // 1. 获取并验证节点和流程 var node = await _sqlSugar.Queryable() .FirstAsync(n => n.Id == nodeId && n.IsDel == 0); if (node == null) return new Result { Code = 400, Msg = "当前节点不存在或已被删除。" }; // 1.2. 用户权限验证 if (!HasConfNodeOperationPermission(node, currUserId)) return new Result { Code = 400, Msg = "当前用户没有操作此节点的权限。" }; var process = await _sqlSugar.Queryable() .FirstAsync(p => p.Id == node.ProcessId && p.IsDel == 0); if (process == null) return new Result { Code = 400, Msg = "当前流程不存在或已被删除。" }; // 2.1 存储更新前流程及节点信息 var nodeBefore = CloneConfNode(node); if (isDtNul) node.ActualDone = actualDone; else node.ActualDone = null; node.IsFileUp = isFileUp; node.Participators = dto.Participators; node.Remark = dto.Remark; // 3. 保存节点更新 await _sqlSugar.Updateable(node) .UpdateColumns(n => new { n.ActualDone, n.Participators, n.IsFileUp, n.Remark }) .ExecuteCommandAsync(); //记录节点日志 await LogConfNodeOpAsync(nodeBefore, node, "Update", currUserId); //当前节点未完成则更改节点状态为已完成 if (node.OverallStatus == ProcessStatus.InProgress && isDtNul) { var statusResult = await UpdateConfNodeStatusSimpleAsync(node.Id, dto.CurrUserId); //日志记录执行结果 _logger.LogInformation($"团组流程设置完成时间:调用更改状态接口(SetActualDoneAsync -> UpdateNodeStatusAsync):[状态变更:进行中 -> 已完成] result: Code={statusResult.Code}, Msg={statusResult.Msg}"); } //当前节点 实际完成时间设置为空且当前状态为已完成 则更改当前节点状态为进行中 else if (node.OverallStatus == ProcessStatus.Completed && !isDtNul) { var statusResult = await UpdateConfNodeStatusSimpleAsync(node.Id, dto.CurrUserId, ProcessStatus.InProgress); //日志记录执行结果 _logger.LogInformation($"团组流程设置完成时间调:用更改状态接口(SetActualDoneAsync -> UpdateNodeStatusAsync):[状态变更:已完成 -> 进行中] result: Code={statusResult.Code}, Msg={statusResult.Msg}"); } return new Result { Code = 200, Msg = "设置成功。" }; } #endregion #region 操作日志 #region 团组流程 /// /// 记录流程操作日志 /// /// 操作前 /// 操作后 /// 操作类型(Create - 创建、Update - 更新、Complete - 完成) /// 操作人ID /// 异步任务 public async Task LogProcessOpAsync(Grp_ProcessOverview? before, Grp_ProcessOverview? after, string opType, int operId) { var chgDetails = GetProcessChgDetails(before, after); var log = new Grp_ProcessLog { ProcessId = after?.Id ?? before?.Id, GroupId = after?.GroupId ?? before?.GroupId ?? 0, OpType = opType, OpDesc = GenerateProcessOpDesc(opType, before, after, chgDetails), BeforeData = before != null ? JsonConvert.SerializeObject(before, GetJsonSettings()) : null, AfterData = after != null ? JsonConvert.SerializeObject(after, GetJsonSettings()) : null, ChgFields = string.Join(",", chgDetails.Select(x => x.FieldName)), CreateUserId = operId }; await _sqlSugar.Insertable(log).ExecuteCommandAsync(); } /// /// 记录节点操作日志 /// /// 操作前 /// 操作后 /// 操作类型(Create - 创建、Update - 更新、Start - 启动、Complete - 完成) /// 操作人ID /// 异步任务 public async Task LogNodeOpAsync(Grp_ProcessNode? before, Grp_ProcessNode? after, string opType, int operId) { var chgDetails = GetNodeChgDetails(before, after); var log = new Grp_ProcessLog { NodeId = after?.Id ?? before?.Id, ProcessId = after?.ProcessId ?? before?.ProcessId, GroupId = 0, // 通过流程ID关联获取 OpType = opType, OpDesc = GenerateNodeOpDesc(opType, before, after, chgDetails), BeforeData = before != null ? JsonConvert.SerializeObject(before, GetJsonSettings()) : null, AfterData = after != null ? JsonConvert.SerializeObject(before, GetJsonSettings()) : null, ChgFields = string.Join(",", chgDetails.Select(x => x.FieldName)), CreateUserId = operId }; await _sqlSugar.Insertable(log).ExecuteCommandAsync(); } /// /// 获取流程变更详情 /// /// 变更前 /// 变更后 /// 变更详情 private List GetProcessChgDetails(Grp_ProcessOverview before, Grp_ProcessOverview after) { var chgDetails = new List(); if (before == null || after == null) return chgDetails; var props = typeof(Grp_ProcessOverview).GetProperties(BindingFlags.Public | BindingFlags.Instance) .Where(p => p.CanRead && p.CanWrite && !IsExclField(p.Name)); foreach (var prop in props) { var beforeVal = prop.GetValue(before); var afterVal = prop.GetValue(after); if (!Equals(beforeVal, afterVal)) { chgDetails.Add(new FieldChgDetail { FieldName = prop.Name, BeforeValue = FormatVal(beforeVal), AfterValue = FormatVal(afterVal) }); } } return chgDetails; } /// /// 获取节点变更详情 /// /// 变更前 /// 变更后 /// 变更详情 private List GetNodeChgDetails(Grp_ProcessNode before, Grp_ProcessNode after) { var chgDetails = new List(); if (before == null || after == null) return chgDetails; var props = typeof(Grp_ProcessNode).GetProperties(BindingFlags.Public | BindingFlags.Instance) .Where(p => p.CanRead && p.CanWrite && !IsExclField(p.Name)); foreach (var prop in props) { var beforeVal = prop.GetValue(before); var afterVal = prop.GetValue(after); if (!Equals(beforeVal, afterVal)) { chgDetails.Add(new FieldChgDetail { FieldName = prop.Name, BeforeValue = FormatVal(beforeVal), AfterValue = FormatVal(afterVal) }); } } return chgDetails; } /// /// 生成流程操作描述 /// /// 操作类型 /// 操作前 /// 操作后 /// 变更详情 /// 操作描述 private string GenerateProcessOpDesc(string opType, Grp_ProcessOverview before, Grp_ProcessOverview after, List chgDetails) { var processType = after?.ProcessType ?? before?.ProcessType; var processName = GetProcessTypeName(processType); if (!chgDetails.Any()) { return opType switch { "Create" => $"创建流程:{processName}", "Update" => $"更新流程:{processName} - 无变更", "Start" => $"启动流程:{processName}", "Complete" => $"完成流程:{processName}", "Delete" => $"删除流程:{processName}", _ => $"{opType}:{processName}" }; } var chgDesc = string.Join("; ", chgDetails.Select(x => $"{GetFieldDisplayName(x.FieldName)} ({x.BeforeValue} -> {x.AfterValue})")); return $"{GetOpTypeDisplay(opType)}:{processName} - {chgDesc}"; } /// /// 生成节点操作描述 /// /// 操作类型 /// 操作前 /// 操作后 /// 变更详情 /// 操作描述 private string GenerateNodeOpDesc(string opType, Grp_ProcessNode before, Grp_ProcessNode after, List chgDetails) { var nodeName = after?.NodeName ?? before?.NodeName; if (!chgDetails.Any()) { return opType switch { "Create" => $"创建节点:{nodeName}", "Update" => $"更新节点:{nodeName} - 无变更", "Start" => $"启动节点:{nodeName}", "Complete" => $"完成节点:{nodeName}", //"Delete" => $"删除节点:{nodeName}", _ => $"{opType}:{nodeName}" }; } var chgDesc = string.Join("; ", chgDetails.Select(x => $"{GetFieldDisplayName(x.FieldName)} ({x.BeforeValue} -> {x.AfterValue})")); return $"{GetOpTypeDisplay(opType)}:{nodeName} - {chgDesc}"; } /// /// 获取流程类型名称 /// /// 流程类型 /// 流程名称 private static string GetProcessTypeName(GroupProcessType? processType) { return processType switch { GroupProcessType.Invitation => "商邀报批", GroupProcessType.Visa => "签证", GroupProcessType.AirTicket => "机票", GroupProcessType.Hotel => "酒店", GroupProcessType.LocalGuide => "地接", GroupProcessType.FeeSettle => "费用结算", _ => "未知流程" }; } /// /// 获取流程日志 /// /// 流程ID /// 日志列表 public async Task> GetProcessLogsAsync(int processId) { return await _sqlSugar.Queryable() .Where(x => x.ProcessId == processId) .OrderByDescending(x => x.CreateTime) .ToListAsync(); } /// /// 获取团组流程日志 /// /// 团组ID /// 日志列表 public async Task> GetGroupLogsAsync(int groupId) { return await _sqlSugar.Queryable() .Where(x => x.GroupId == groupId) .OrderByDescending(x => x.CreateTime) .ToListAsync(); } #endregion #region 会务流程 /// /// 记录会务流程操作日志 /// /// 操作前 /// 操作后 /// 操作类型(Create - 创建、Update - 更新、Complete - 完成) /// 操作人ID /// 异步任务 public async Task LogConfProcessOpAsync(Grp_ConfProcessOverview? before, Grp_ConfProcessOverview? after, string opType, int operId) { var chgDetails = GetConfProcessChgDetails(before, after); var log = new Grp_ConfProcessLog { ProcessId = after?.Id ?? before?.Id, GroupId = after?.GroupId ?? before?.GroupId ?? 0, OpType = opType, OpDesc = GenerateConfProcessOpDesc(opType, before, after, chgDetails), BeforeData = before != null ? JsonConvert.SerializeObject(before, GetJsonSettings()) : null, AfterData = after != null ? JsonConvert.SerializeObject(after, GetJsonSettings()) : null, ChgFields = string.Join(",", chgDetails.Select(x => x.FieldName)), CreateUserId = operId }; await _sqlSugar.Insertable(log).ExecuteCommandAsync(); } /// /// 记录会务节点操作日志 /// /// 操作前 /// 操作后 /// 操作类型(Create - 创建、Update - 更新、Start - 启动、Complete - 完成) /// 操作人ID /// 异步任务 public async Task LogConfNodeOpAsync(Grp_ConfProcessNode? before, Grp_ConfProcessNode? after, string opType, int operId) { var chgDetails = GetConfNodeChgDetails(before, after); var log = new Grp_ConfProcessLog { NodeId = after?.Id ?? before?.Id, ProcessId = after?.ProcessId ?? before?.ProcessId, GroupId = 0, // 通过流程ID关联获取 OpType = opType, OpDesc = GenerateConfNodeOpDesc(opType, before, after, chgDetails), BeforeData = before != null ? JsonConvert.SerializeObject(before, GetJsonSettings()) : null, AfterData = after != null ? JsonConvert.SerializeObject(before, GetJsonSettings()) : null, ChgFields = string.Join(",", chgDetails.Select(x => x.FieldName)), CreateUserId = operId }; await _sqlSugar.Insertable(log).ExecuteCommandAsync(); } /// /// 获取流程变更详情 /// /// 变更前 /// 变更后 /// 变更详情 private static List GetConfProcessChgDetails(Grp_ConfProcessOverview before, Grp_ConfProcessOverview after) { var chgDetails = new List(); if (before == null || after == null) return chgDetails; var props = typeof(Grp_ConfProcessOverview).GetProperties(BindingFlags.Public | BindingFlags.Instance) .Where(p => p.CanRead && p.CanWrite && !IsExclField(p.Name)); foreach (var prop in props) { var beforeVal = prop.GetValue(before); var afterVal = prop.GetValue(after); if (!Equals(beforeVal, afterVal)) { chgDetails.Add(new FieldChgDetail { FieldName = prop.Name, BeforeValue = FormatVal(beforeVal), AfterValue = FormatVal(afterVal) }); } } return chgDetails; } /// /// 获取节点变更详情 /// /// 变更前 /// 变更后 /// 变更详情 private static List GetConfNodeChgDetails(Grp_ConfProcessNode before, Grp_ConfProcessNode after) { var chgDetails = new List(); if (before == null || after == null) return chgDetails; var props = typeof(Grp_ConfProcessNode).GetProperties(BindingFlags.Public | BindingFlags.Instance) .Where(p => p.CanRead && p.CanWrite && !IsExclField(p.Name)); foreach (var prop in props) { var beforeVal = prop.GetValue(before); var afterVal = prop.GetValue(after); if (!Equals(beforeVal, afterVal)) { chgDetails.Add(new FieldChgDetail { FieldName = prop.Name, BeforeValue = FormatVal(beforeVal), AfterValue = FormatVal(afterVal) }); } } return chgDetails; } /// /// 生成流程操作描述 /// /// 操作类型 /// 操作前 /// 操作后 /// 变更详情 /// 操作描述 private static string GenerateConfProcessOpDesc(string opType, Grp_ConfProcessOverview before, Grp_ConfProcessOverview after, List chgDetails) { var processType = after?.ProcessType ?? before?.ProcessType; var processName = GetConfProcessTypeName(processType); if (!chgDetails.Any()) { return opType switch { "Create" => $"创建流程:{processName}", "Update" => $"更新流程:{processName} - 无变更", "Start" => $"启动流程:{processName}", "Complete" => $"完成流程:{processName}", "Delete" => $"删除流程:{processName}", _ => $"{opType}:{processName}" }; } var chgDesc = string.Join("; ", chgDetails.Select(x => $"{GetFieldDisplayName(x.FieldName)} ({x.BeforeValue} -> {x.AfterValue})")); return $"{GetOpTypeDisplay(opType)}:{processName} - {chgDesc}"; } /// /// 生成节点操作描述 /// /// 操作类型 /// 操作前 /// 操作后 /// 变更详情 /// 操作描述 private static string GenerateConfNodeOpDesc(string opType, Grp_ConfProcessNode before, Grp_ConfProcessNode after, List chgDetails) { var nodeName = after?.NodeName ?? before?.NodeName; if (!chgDetails.Any()) { return opType switch { "Create" => $"创建节点:{nodeName}", "Update" => $"更新节点:{nodeName} - 无变更", "Start" => $"启动节点:{nodeName}", "Complete" => $"完成节点:{nodeName}", //"Delete" => $"删除节点:{nodeName}", _ => $"{opType}:{nodeName}" }; } var chgDesc = string.Join("; ", chgDetails.Select(x => $"{GetFieldDisplayName(x.FieldName)} ({x.BeforeValue} -> {x.AfterValue})")); return $"{GetOpTypeDisplay(opType)}:{nodeName} - {chgDesc}"; } /// /// 获取流程类型名称 /// /// 流程类型 /// 流程名称 private static string GetConfProcessTypeName(ConfProcessType? processType) { return processType switch { ConfProcessType.Conference => "会务", _ => "未知流程" }; } /// /// 获取会务流程日志 /// /// 流程ID /// 日志列表 public async Task> GetConfProcessLogsAsync(int processId) { return await _sqlSugar.Queryable() .Where(x => x.ProcessId == processId) .OrderByDescending(x => x.CreateTime) .ToListAsync(); } /// /// 获取团组会务流程日志 /// /// 团组ID /// 日志列表 public async Task> GetGroupConfLogsAsync(int groupId) { return await _sqlSugar.Queryable() .Where(x => x.GroupId == groupId) .OrderByDescending(x => x.CreateTime) .ToListAsync(); } #endregion #region 日志私有方法 /// /// 获取JSON序列化设置 /// /// JSON设置 private static JsonSerializerSettings GetJsonSettings() { return new JsonSerializerSettings { ReferenceLoopHandling = ReferenceLoopHandling.Ignore, NullValueHandling = NullValueHandling.Ignore, DateFormatString = "yyyy-MM-dd HH:mm:ss", Formatting = Formatting.None }; } /// /// 获取操作类型显示 /// /// 操作类型 /// 显示名称 private static string GetOpTypeDisplay(string opType) { return opType switch { "Create" => "创建", "Update" => "更新", "Start" => "启动", "Complete" => "完成", "Delete" => "删除", "StatusChg" => "状态变更", _ => opType }; } /// /// 获取字段显示名称 /// /// 字段名 /// 显示名称 private static string GetFieldDisplayName(string fieldName) { return fieldName switch { "OverallStatus" => "状态", "ProcessOrder" => "流程顺序", "StartTime" => "开始时间", "EndTime" => "结束时间", "UpdatedUserId" => "更新人", "UpdatedTime" => "更新时间", "NodeOrder" => "节点顺序", "NodeName" => "节点名称", "NodeDescTips" => "节点描述提示", "IsCurrent" => "当前节点", "Participator" => "参与人", "Operator" => "操作人", "OperationTime" => "操作时间", "ActualDone" => "实际完成时间", "IsFileUp" => "是否上传文件(签证、机票、酒店、地接 流程结尾节点使用)", "IsAssist" => "是否协助(财务流程首节点使用)", "IsPart" => "是否参与(商邀 第五步使用)", _ => fieldName }; } /// /// 格式化值显示 /// /// 值 /// 格式化值 private static string FormatVal(object value) { if (value == null) return "空"; if (value is ProcessStatus status) { return status switch { ProcessStatus.UnStarted => "未开始", ProcessStatus.InProgress => "进行中", ProcessStatus.Completed => "已完成", _ => status.ToString() }; } if (value is bool boolVal) return boolVal ? "是" : "否"; if (value is DateTime dateVal) return dateVal.ToString("yyyy-MM-dd HH:mm"); var strVal = value.ToString(); return string.IsNullOrEmpty(strVal) ? "空" : strVal; } /// /// 检查是否排除字段 /// /// 字段名 /// 是否排除 private static bool IsExclField(string fieldName) { var exclFields = new List { "Id", "CreateTime", "CreateUserId", "Nodes", "Process" // 导航属性 }; return exclFields.Contains(fieldName); } #endregion #endregion } }