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
}
}