ソースを参照

Merge branch 'release'

xf 5 ヶ月 前
コミット
2a69660ff6
71 ファイル変更2823 行追加1137 行削除
  1. 1 1
      src/Hotline.Api/Controllers/Bi/BiCallController.cs
  2. 269 108
      src/Hotline.Api/Controllers/Bi/BiOrderController.cs
  3. 6 18
      src/Hotline.Api/Controllers/CommonPController.cs
  4. 5 2
      src/Hotline.Api/Controllers/ExportWordController.cs
  5. 61 7
      src/Hotline.Api/Controllers/KnowledgeController.cs
  6. 308 385
      src/Hotline.Api/Controllers/OrderController.cs
  7. 1 1
      src/Hotline.Api/Controllers/OrderRevocationController.cs
  8. 1 1
      src/Hotline.Api/config/appsettings.Development.json
  9. 13 0
      src/Hotline.Application.Contracts/Validators/Order/OrderBatchFileDtoValidator.cs
  10. 16 3
      src/Hotline.Application.Tests/Application/KnowApplicationTest.cs
  11. 3 56
      src/Hotline.Application.Tests/Application/OrderApplicationTest.cs
  12. 5 4
      src/Hotline.Application.Tests/Application/ZiGongCallReportApplicationTest.cs
  13. 10 5
      src/Hotline.Application.Tests/Controller/DefaultHttpContextAccessor.cs
  14. 70 0
      src/Hotline.Application.Tests/Controller/KnowledgeControllerTest.cs
  15. 108 20
      src/Hotline.Application.Tests/Controller/OrderControllerTest.cs
  16. 1 1
      src/Hotline.Application.Tests/Controller/PushMessageControllerTest.cs
  17. 3 2
      src/Hotline.Application.Tests/Domain/OrderVisitDomainServiceTest.cs
  18. 12 28
      src/Hotline.Application.Tests/Domain/YiBinExpireTimeTest.cs
  19. 26 55
      src/Hotline.Application.Tests/Domain/ZiGongExpireTimeTest.cs
  20. 14 0
      src/Hotline.Application.Tests/Dto/CreateOrderOutDto.cs
  21. 0 4
      src/Hotline.Application.Tests/Hotline.Application.Tests.csproj
  22. 4 0
      src/Hotline.Application.Tests/Infrastructure/TestSettingConstants.cs
  23. 53 0
      src/Hotline.Application.Tests/Mock/KnowledgeServiceMock.cs
  24. 239 0
      src/Hotline.Application.Tests/Mock/OrderServiceMock.cs
  25. 25 0
      src/Hotline.Application.Tests/Startup.cs
  26. 28 13
      src/Hotline.Application.Tests/TestBase.cs
  27. 16 5
      src/Hotline.Application/ExportWord/WordHelper.cs
  28. 0 6
      src/Hotline.Application/FlowEngine/IWorkflowApplication.cs
  29. 17 92
      src/Hotline.Application/FlowEngine/WorkflowApplication.cs
  30. 1 0
      src/Hotline.Application/Handlers/FlowEngine/WorkflowEndHandler.cs
  31. 4 0
      src/Hotline.Application/Hotline.Application.csproj
  32. 1 0
      src/Hotline.Application/Mappers/MapperConfigs.cs
  33. 19 10
      src/Hotline.Application/Mappers/OrderMapperConfigs.cs
  34. 18 2
      src/Hotline.Application/Orders/IOrderApplication.cs
  35. 337 46
      src/Hotline.Application/Orders/OrderApplication.cs
  36. 1 1
      src/Hotline.Application/Subscribers/DatasharingSubscriber.cs
  37. 14 2
      src/Hotline.Repository.SqlSugar/Orders/OrderRepository.cs
  38. 47 0
      src/Hotline.Repository.SqlSugar/System/SystemLogRepository.cs
  39. 10 0
      src/Hotline.Share/Dtos/Bi/BiOrderDto.cs
  40. 1 1
      src/Hotline.Share/Dtos/CallCenter/TelDto.cs
  41. 2 1
      src/Hotline.Share/Dtos/FlowEngine/NextStepOption.cs
  42. 10 0
      src/Hotline.Share/Dtos/FlowEngine/NextStepsDto.cs
  43. 23 0
      src/Hotline.Share/Dtos/Knowledge/KnowledgeDto.cs
  44. 18 0
      src/Hotline.Share/Dtos/Order/Handle/OrderBatchFileDto.cs
  45. 8 2
      src/Hotline.Share/Dtos/Order/OrderBiDto.cs
  46. 63 13
      src/Hotline.Share/Dtos/Order/OrderDto.cs
  47. 3 2
      src/Hotline.Share/Dtos/Order/OrderStartFlowDto.cs
  48. 13 2
      src/Hotline.Share/Dtos/Order/OrderVisitDto.cs
  49. 11 1
      src/Hotline.Share/Dtos/Order/OrderWaitedDto.cs
  50. 10 0
      src/Hotline.Share/Dtos/Order/PublishedDto.cs
  51. 12 1
      src/Hotline.Share/Dtos/Order/QueryOrderDto.cs
  52. 5 0
      src/Hotline.Share/Dtos/Settings/TimeConfig.cs
  53. 128 0
      src/Hotline.Share/Dtos/WebPortal/GetCaseReultSendModel.cs
  54. 10 0
      src/Hotline.Share/Requests/DepartmentalProcessingStatisticsDto.cs
  55. 25 0
      src/Hotline.Share/Requests/PagedKeywordRequest.cs
  56. 43 0
      src/Hotline.Share/Tools/HtmlImageValidator.cs
  57. 5 0
      src/Hotline.Share/Tools/ListExtensions.cs
  58. 5 0
      src/Hotline/Caching/Interfaces/ISystemSettingCacheManager.cs
  59. 32 1
      src/Hotline/Caching/Services/SystemSettingCacheManager.cs
  60. 7 17
      src/Hotline/FlowEngine/Workflows/IWorkflowDomainService.cs
  61. 10 0
      src/Hotline/FlowEngine/Workflows/StepBasicEntity.cs
  62. 38 32
      src/Hotline/FlowEngine/Workflows/Workflow.cs
  63. 240 163
      src/Hotline/FlowEngine/Workflows/WorkflowDomainService.cs
  64. 5 13
      src/Hotline/FlowEngine/Workflows/WorkflowStep.cs
  65. 2 2
      src/Hotline/Orders/IOrderDomainService.cs
  66. 33 4
      src/Hotline/Orders/Order.cs
  67. 207 2
      src/Hotline/Orders/OrderDomainService.cs
  68. 21 0
      src/Hotline/Settings/ISystemLogRepository.cs
  69. 10 0
      src/Hotline/Settings/SettingConstants.cs
  70. 54 0
      src/Hotline/dataview.md
  71. 2 2
      src/XF.Domain/Extensions/StringExtensions.cs

+ 1 - 1
src/Hotline.Api/Controllers/Bi/BiCallController.cs

@@ -165,7 +165,7 @@ public class BiCallController : BaseController
     /// <param name="dto"></param>
     /// <returns></returns>
     [HttpGet("query_calls_statistics_detail/base-data")]
-    public async Task<Dictionary<string, dynamic>> QueryCallsStatisticsDetailBaseData()
+    public Dictionary<string, dynamic> QueryCallsStatisticsDetailBaseData()
     {
         return _baseDataApplication
             .EndBy()

+ 269 - 108
src/Hotline.Api/Controllers/Bi/BiOrderController.cs

@@ -27,6 +27,7 @@ using Hotline.Share.Requests;
 using Hotline.Tools;
 using MapsterMapper;
 using Microsoft.AspNetCore.Mvc;
+using Org.BouncyCastle.Utilities.Collections;
 using SqlSugar;
 using System.Data;
 using XF.Domain.Authentications;
@@ -538,6 +539,121 @@ namespace Hotline.Api.Controllers.Bi
             return new PagedDto<HotspotDataLsitVo>(total, items);
         }
 
+        /// <summary>
+        /// 热点数据统计导出
+        /// </summary>
+        /// <param name="dto"></param>
+        /// <returns></returns>
+        [HttpPost("hotspot_data_list/export")]
+        public async Task<FileStreamResult> HotspotDataLsitExprot([FromBody] ExportExcelDto<HotspotReportPagedRequest> dto)
+        {
+            if (!dto.QueryDto.StartTime.HasValue || !dto.QueryDto.EndTime.HasValue) throw UserFriendlyException.SameMessage("请选择时间!");
+            if (dto.QueryDto.Type == 0 && (!dto.QueryDto.ChainStartTime.HasValue || !dto.QueryDto.ChainEndTime.HasValue)) throw UserFriendlyException.SameMessage("请选择环比时间!");
+
+            dto.QueryDto.EndTime = dto.QueryDto.EndTime.Value.AddDays(1).AddSeconds(-1);
+
+            var IsCenter = _sessionContext.OrgIsCenter;
+
+            if (dto.QueryDto.Type == 0)
+            {
+                dto.QueryDto.ChainEndTime = dto.QueryDto.ChainEndTime.Value.AddDays(1).AddSeconds(-1);
+            }
+
+            var items = await _hotspotTypeRepository.Queryable(false, true)
+                .LeftJoin<Order>((x, o) => o.HotspotSpliceName != null && (x.HotSpotName == o.HotspotSpliceName || o.HotspotSpliceName.Contains(x.HotSpotName)) && o.IsDeleted == false)
+                .WhereIF(dto.QueryDto.StartTime.HasValue, (x, o) => o.CreationTime >= dto.QueryDto.StartTime)
+                .WhereIF(dto.QueryDto.EndTime.HasValue, (x, o) => o.CreationTime <= dto.QueryDto.EndTime)
+                .WhereIF(!string.IsNullOrEmpty(dto.QueryDto.Keyword), (x, o) => x.HotSpotName.Contains(dto.QueryDto.Keyword!))
+                .WhereIF(IsCenter == false, (x, o) => o.ActualHandleOrgCode == _sessionContext.RequiredOrgId)
+                .Where((x, o) => x.ParentId == dto.QueryDto.Id)
+                .Where((x, o) => x.IsDeleted == false)
+                .GroupBy((x, o) => new { x.Id, x.HotSpotName })
+                .Select((x, o) => new HotspotDataLsitVo
+                {
+                    Id = x.Id,
+                    Name = x.HotSpotName,
+                    Num = SqlFunc.AggregateSum(SqlFunc.IIF(o.Id != null, 1, 0)),
+                    Sublevel = SqlFunc.AggregateSum(SqlFunc.IIF(x.HotSpotName != o.HotspotName, 1, 0)) > 0,
+                }).MergeTable().ToListAsync();
+            var chainStartTime = dto.QueryDto.StartTime;
+            var chainEndTime = dto.QueryDto.EndTime;
+            switch (dto.QueryDto.Type)
+            {
+                case 1://日
+                    chainStartTime = dto.QueryDto.StartTime.Value.AddDays(-1);
+                    chainEndTime = dto.QueryDto.EndTime.Value.AddDays(-1);
+                    break;
+                case 2://月
+                    chainStartTime = dto.QueryDto.StartTime.Value.AddMonths(-1);
+                    chainEndTime = dto.QueryDto.EndTime.Value.AddMonths(-1);
+                    break;
+                case 3://年
+                    chainStartTime = dto.QueryDto.StartTime.Value.AddYears(-1);
+                    chainEndTime = dto.QueryDto.EndTime.Value.AddYears(-1);
+                    break;
+                case 0:
+                    chainStartTime = dto.QueryDto.ChainStartTime.Value;
+                    chainEndTime = dto.QueryDto.ChainEndTime.Value;
+                    break;
+            }
+            var chainItems = await _hotspotTypeRepository.Queryable(false, true)
+                .LeftJoin<Order>((x, o) => o.HotspotSpliceName != null && (x.HotSpotName == o.HotspotSpliceName || o.HotspotSpliceName.Contains(x.HotSpotName)) && o.IsDeleted == false)
+                .WhereIF(dto.QueryDto.StartTime.HasValue, (x, o) => o.CreationTime >= chainStartTime)
+                .WhereIF(dto.QueryDto.EndTime.HasValue, (x, o) => o.CreationTime <= chainEndTime)
+                .WhereIF(!string.IsNullOrEmpty(dto.QueryDto.Keyword), (x, o) => x.HotSpotName.Contains(dto.QueryDto.Keyword!))
+                .WhereIF(IsCenter == false, (x, o) => o.ActualHandleOrgCode == _sessionContext.RequiredOrgId)
+                .Where((x, o) => x.ParentId == dto.QueryDto.Id)
+                .Where((x, o) => x.IsDeleted == false)
+                .GroupBy((x, o) => new { x.Id, x.HotSpotName })
+                .Select((x, o) => new
+                {
+                    Id = x.Id,
+                    ChainNum = SqlFunc.AggregateSum(SqlFunc.IIF(o.Id != null, 1, 0)),
+                }).MergeTable().ToListAsync();
+            var res = (from t1 in items
+                       join t2 in chainItems on t1.Id equals t2.Id into t1_t2
+                       from item in t1_t2.DefaultIfEmpty()
+                       select new HotspotDataLsitVo
+                       {
+                           Id = t1.Id,
+                           Name = t1.Name,
+                           Num = t1.Num,
+                           Sublevel = t1.Sublevel,
+                           Children = new List<HotspotDataLsitVo>(),
+                           ChainNum = t1_t2.Select(x => x.ChainNum).FirstOrDefault(),
+                           ChainRate = t1_t2.Select(x => x.ChainNum).FirstOrDefault() > 0 ?
+                           ((double.Parse(t1.Num.ToString()) - double.Parse(t1_t2.Select(x => x.ChainNum).FirstOrDefault().ToString())) / double.Parse(t1_t2.Select(x => x.ChainNum).FirstOrDefault().ToString()) * 100).ToString("F2") + "%" : "100.00%",
+                       }).ToList();
+            var total = new HotspotDataLsitVo()
+            {
+                Id = "0",
+                Name = "合计",
+                Num = res.Sum(x => x.Num),
+                Sublevel = false,
+                Children = new List<HotspotDataLsitVo>(),
+                ChainNum = res.Sum(x => x.ChainNum),
+                ChainRate = res.Sum(x => x.ChainNum) > 0 ? ((double.Parse(res.Sum(x => x.Num).ToString()) - double.Parse(res.Sum(x => x.ChainNum).ToString())) / double.Parse(res.Sum(x => x.ChainNum).ToString()) * 100).ToString("F2") + "%" : "100.00%"
+            };
+            res.Add(total);
+
+            List<HotspotDataLsitVo> data;
+            data = res;
+
+            var dataDtos = _mapper.Map<ICollection<HotspotDataLsitVo>>(data);
+
+            dynamic? dynamicClass = DynamicClassHelper.CreateDynamicClass(dto.ColumnInfos);
+
+            var dtos = dataDtos
+                .Select(stu => _mapper.Map(stu, typeof(HotspotDataLsitVo), dynamicClass))
+                .Cast<object>()
+                .ToList();
+
+            var stream = ExcelHelper.CreateStream(dtos);
+
+            return ExcelStreamResult(stream, "热点类型统计");
+            //return new { List = res, Total = total };
+        }
+
         /// <summary>
         /// 热点数据统计 TODO和前端沟通
         /// </summary>
@@ -612,7 +728,7 @@ namespace Hotline.Api.Controllers.Bi
             var res = (from t1 in items
                        join t2 in chainItems on t1.Id equals t2.Id into t1_t2
                        from item in t1_t2.DefaultIfEmpty()
-                       select new
+                       select new HotspotDataLsitVo()
                        {
                            Id = t1.Id,
                            Name = t1.Name,
@@ -623,7 +739,7 @@ namespace Hotline.Api.Controllers.Bi
                            ChainRate = t1_t2.Select(x => x.ChainNum).FirstOrDefault() > 0 ?
                            ((double.Parse(t1.Num.ToString()) - double.Parse(t1_t2.Select(x => x.ChainNum).FirstOrDefault().ToString())) / double.Parse(t1_t2.Select(x => x.ChainNum).FirstOrDefault().ToString()) * 100).ToString("F2") + "%" : "100.00%",
                        }).ToList();
-            var total = new
+            var total = new HotspotDataLsitVo()
             {
                 Id = "0",
                 Name = "合计",
@@ -892,84 +1008,34 @@ namespace Hotline.Api.Controllers.Bi
             return new PagedDto<OrderSpecialDto>(total, _mapper.Map<IReadOnlyList<OrderSpecialDto>>(items));
         }
 
-        /// <summary>
-        /// 受理类型前十
-        /// </summary>
-        /// <param name="dto"></param>
-        /// <returns></returns>
-        [HttpGet("accept_type_top10_list")]
-        public async Task<PagedDto<AcceptTypeTop10Vo>> AcceptTypeTop10List([FromQuery] ReportPagedRequest dto)
+
+		/// <summary>
+		/// 受理类型前十
+		/// </summary>
+		/// <param name="dto"></param>
+		/// <returns></returns>
+		[HttpGet("accept_type_top10_list")]
+        public async Task<object> AcceptTypeTop10List([FromQuery] ReportPagedRequest dto)
         {
-            if (!dto.StartTime.HasValue || !dto.EndTime.HasValue) throw UserFriendlyException.SameMessage("请选择时间!");
+            var (areaList, returnList,data) = await _orderApplication.AcceptTypeTop10List(dto,false);
 
-            dto.PageIndex = 1;
-            dto.PageSize = 10;
-            var IsCenter = _sessionContext.OrgIsCenter;
-            var query = _orderRepository.Queryable(false, false, false)
-                .WhereIF(dto.StartTime.HasValue, x => x.CreationTime >= dto.StartTime)
-                .WhereIF(dto.EndTime.HasValue, x => x.CreationTime <= dto.EndTime)
-                .WhereIF(IsCenter == false, x => x.ActualHandleOrgCode.StartsWith(_sessionContext.RequiredOrgId))
-                .Select(x => new
-                {
-                    AcceptType = x.AcceptType,
-                    OneHotspot = SqlFunc.Substring(x.HotspotSpliceName, 0, SqlFunc.CharIndex("-", x.HotspotSpliceName + "-")),
-                    Id = x.Id
-                }).MergeTable().Where(x => x.OneHotspot != "非受理范围")
-                .GroupBy(x => new { x.OneHotspot })
-                .Select(x => new AcceptTypeTop10Vo
-                {
-                    Name = x.OneHotspot,
-                    ValidAccept = SqlFunc.AggregateSum(SqlFunc.IIF(true, 1, 0)),
-                    Consult = SqlFunc.AggregateSum(SqlFunc.IIF("咨询".Equals(x.AcceptType), 1, 0)),
-                    Report = SqlFunc.AggregateSum(SqlFunc.IIF("举报".Equals(x.AcceptType), 1, 0)),
-                    Complaint = SqlFunc.AggregateSum(SqlFunc.IIF("投诉".Equals(x.AcceptType), 1, 0)),
-                    SeekHelp = SqlFunc.AggregateSum(SqlFunc.IIF("求助".Equals(x.AcceptType), 1, 0)),
-                    Suggest = SqlFunc.AggregateSum(SqlFunc.IIF("建议".Equals(x.AcceptType), 1, 0)),
-                    Opinion = SqlFunc.AggregateSum(SqlFunc.IIF("意见".Equals(x.AcceptType), 1, 0)),
-                    Rests = SqlFunc.AggregateSum(SqlFunc.IIF("其他".Equals(x.AcceptType), 1, 0)),
-                    BenefitThePeople = SqlFunc.AggregateSum(SqlFunc.IIF("惠民帮助".Equals(x.AcceptType), 1, 0)),
-                    Praise = SqlFunc.AggregateSum(SqlFunc.IIF("表扬".Equals(x.AcceptType), 1, 0)),
-                }).MergeTable();
-            switch (dto.SortField)
-            {
-                case "validAccept":
-                    query = dto.SortRule is 0 ? query.OrderBy(x => x.ValidAccept) : query.OrderByDescending(x => x.ValidAccept);
-                    break;
-                case "consult":
-                    query = dto.SortRule is 0 ? query.OrderBy(x => x.Consult) : query.OrderByDescending(x => x.Consult);
-                    break;
-                case "report":
-                    query = dto.SortRule is 0 ? query.OrderBy(x => x.Report) : query.OrderByDescending(x => x.Report);
-                    break;
-                case "complaint":
-                    query = dto.SortRule is 0 ? query.OrderBy(x => x.Complaint) : query.OrderByDescending(x => x.Complaint);
-                    break;
-                case "seekHelp":
-                    query = dto.SortRule is 0 ? query.OrderBy(x => x.SeekHelp) : query.OrderByDescending(x => x.SeekHelp);
-                    break;
-                case "suggest":
-                    query = dto.SortRule is 0 ? query.OrderBy(x => x.Suggest) : query.OrderByDescending(x => x.Suggest);
-                    break;
-                case "opinion":
-                    query = dto.SortRule is 0 ? query.OrderBy(x => x.Opinion) : query.OrderByDescending(x => x.Opinion);
-                    break;
-                case "rests":
-                    query = dto.SortRule is 0 ? query.OrderBy(x => x.Rests) : query.OrderByDescending(x => x.Rests);
-                    break;
-                case "benefitThePeople":
-                    query = dto.SortRule is 0 ? query.OrderBy(x => x.BenefitThePeople) : query.OrderByDescending(x => x.BenefitThePeople);
-                    break;
-                case "praise":
-                    query = dto.SortRule is 0 ? query.OrderBy(x => x.Praise) : query.OrderByDescending(x => x.Praise);
-                    break;
-                default:
-                    query = query.OrderByDescending(x => x.ValidAccept);
-                    break;
-            }
-            var (total, items) = await query.ToPagedListAsync(dto, HttpContext.RequestAborted);
-            return new PagedDto<AcceptTypeTop10Vo>(total, items);
+            return new { List = areaList ,Data = returnList };
         }
 
+		/// <summary>
+		/// 受理类型前十导出
+		/// </summary>
+		/// <param name="dto"></param>
+		/// <returns></returns>
+		[HttpPost("accept_type_top10_list-export")]
+        public async Task<FileStreamResult> AcceptTypeTop10ListExport([FromBody] ReportPagedRequest dto)
+        {
+			var (areaList, returnList,data) = await _orderApplication.AcceptTypeTop10List(dto,true);
+			var stream = ExcelHelper.CreateStream(data);
+            return ExcelStreamResult(stream, "受理类型前十统计");
+        }
+
+
         /// <summary>
         /// 热点类型部门统计
         /// </summary>
@@ -1035,17 +1101,17 @@ namespace Hotline.Api.Controllers.Bi
         /// <param name="TypeId">0:全部 ,1:市民,2:企业</param>
         /// <returns></returns>
         [HttpGet("hotspot-statistics")]
-        public async Task<object> HotspotStatistics(DateTime StartTime, DateTime EndTime, int TypeId, string? HotspotCode)
+        public async Task<object> HotspotStatistics([FromQuery] HotspotStatisticsRep dto)
         {
             var IsCenter = _sessionContext.OrgIsCenter;
 
-            if (string.IsNullOrEmpty(HotspotCode))
+            if (string.IsNullOrEmpty(dto.HotspotCode))
             {
                 var list = await _hotspotTypeRepository.Queryable()
                 .LeftJoin<Order>((it, o) => it.Id == o.HotspotId)
-                .Where((it, o) => o.CreationTime >= StartTime && o.CreationTime <= EndTime && o.Id != null)
-                .WhereIF(TypeId == 1, (it, o) => o.IdentityType == EIdentityType.Citizen)
-                .WhereIF(TypeId == 2, (it, o) => o.IdentityType == EIdentityType.Enterprise)
+                .Where((it, o) => o.CreationTime >= dto.StartTime && o.CreationTime <= dto.EndTime && o.Id != null)
+                .WhereIF(dto.TypeId == 1, (it, o) => o.IdentityType == EIdentityType.Citizen)
+                .WhereIF(dto.TypeId == 2, (it, o) => o.IdentityType == EIdentityType.Enterprise)
                 .WhereIF(IsCenter == false, (it, o) => o.ActualHandleOrgCode.StartsWith(_sessionContext.RequiredOrgId))
                 .GroupBy((it, o) => new { Id = it.Id.Substring(SqlFunc.MappingColumn<int>("0"), SqlFunc.MappingColumn<int>("2")) })
                 .Select((it, o) => new
@@ -1068,13 +1134,13 @@ namespace Hotline.Api.Controllers.Bi
             }
             else
             {
-                string count = (HotspotCode.Length + 2).ToString();
-                string countx = HotspotCode.Length.ToString();
+                string count = (dto.HotspotCode.Length + 2).ToString();
+                string countx = dto.HotspotCode.Length.ToString();
                 var list = await _hotspotTypeRepository.Queryable()
                 .LeftJoin<Order>((it, o) => it.Id == o.HotspotId)
-                .Where((it, o) => o.CreationTime >= StartTime && o.CreationTime <= EndTime && it.ParentId.Substring(SqlFunc.MappingColumn<int>("0"), SqlFunc.MappingColumn<int>(countx)) == HotspotCode)
-                .WhereIF(TypeId == 1, (it, o) => o.IdentityType == EIdentityType.Citizen)
-                .WhereIF(TypeId == 2, (it, o) => o.IdentityType == EIdentityType.Enterprise)
+                .Where((it, o) => o.CreationTime >= dto.StartTime && o.CreationTime <= dto.EndTime && it.ParentId.Substring(SqlFunc.MappingColumn<int>("0"), SqlFunc.MappingColumn<int>(countx)) == dto.HotspotCode)
+                .WhereIF(dto.TypeId == 1, (it, o) => o.IdentityType == EIdentityType.Citizen)
+                .WhereIF(dto.TypeId == 2, (it, o) => o.IdentityType == EIdentityType.Enterprise)
                 .WhereIF(IsCenter == false, (it, o) => o.ActualHandleOrgCode.StartsWith(_sessionContext.RequiredOrgId))
                 .GroupBy((it, o) => new { Id = it.Id.Substring(SqlFunc.MappingColumn<int>("0"), SqlFunc.MappingColumn<int>(count)) })
                 .Select((it, o) => new
@@ -1097,12 +1163,92 @@ namespace Hotline.Api.Controllers.Bi
             }
         }
 
-        /// <summary>
-        /// 部门满意度统计
-        /// </summary>
-        /// <param name="dto"></param>
-        /// <returns></returns>
-        [HttpGet("visit-org-satisfaction-statistics")]
+		/// <summary>
+		/// 热点类型小类统计导出
+		/// </summary>
+		/// <param name="StartTime"></param>
+		/// <param name="EndTime"></param>
+		/// <param name="TypeId">0:全部 ,1:市民,2:企业</param>
+		/// <returns></returns>
+		[HttpPost("hotspot-statistics/export")]
+		public async Task<FileStreamResult> HotspotStatisticsExprot([FromBody] ExportExcelDto<HotspotStatisticsRep> dto)
+		{
+			var IsCenter = _sessionContext.OrgIsCenter;
+            DataTable data = new DataTable();
+
+			if (string.IsNullOrEmpty(dto.QueryDto.HotspotCode))
+			{
+				data = await _hotspotTypeRepository.Queryable()
+				.LeftJoin<Order>((it, o) => it.Id == o.HotspotId)
+				.Where((it, o) => o.CreationTime >= dto.QueryDto.StartTime && o.CreationTime <= dto.QueryDto.EndTime && o.Id != null)
+				.WhereIF(dto.QueryDto.TypeId == 1, (it, o) => o.IdentityType == EIdentityType.Citizen)
+				.WhereIF(dto.QueryDto.TypeId == 2, (it, o) => o.IdentityType == EIdentityType.Enterprise)
+				.WhereIF(IsCenter == false, (it, o) => o.ActualHandleOrgCode.StartsWith(_sessionContext.RequiredOrgId))
+				.GroupBy((it, o) => new { Id = it.Id.Substring(SqlFunc.MappingColumn<int>("0"), SqlFunc.MappingColumn<int>("2")) })
+				.Select((it, o) => new
+				{
+					HotspotCode = it.Id.Substring(SqlFunc.MappingColumn<int>("0"), SqlFunc.MappingColumn<int>("2")),
+					SumCount = SqlFunc.AggregateCount(it.HotSpotName)
+				})
+				.MergeTable()
+				.LeftJoin<Hotspot>((x, q) => x.HotspotCode == q.Id)
+				.Select((x, q) => new
+				{
+					SumCount = x.SumCount,
+					HotspotName = q.HotSpotName
+				})
+				.ToDataTableAsync();
+			}
+			else
+			{
+				string count = (dto.QueryDto.HotspotCode.Length + 2).ToString();
+				string countx = dto.QueryDto.HotspotCode.Length.ToString();
+				 data = await _hotspotTypeRepository.Queryable()
+				.LeftJoin<Order>((it, o) => it.Id == o.HotspotId)
+				.Where((it, o) => o.CreationTime >= dto.QueryDto.StartTime && o.CreationTime <= dto.QueryDto.EndTime && it.ParentId.Substring(SqlFunc.MappingColumn<int>("0"), SqlFunc.MappingColumn<int>(countx)) == dto.QueryDto.HotspotCode)
+				.WhereIF(dto.QueryDto.TypeId == 1, (it, o) => o.IdentityType == EIdentityType.Citizen)
+				.WhereIF(dto.QueryDto.TypeId == 2, (it, o) => o.IdentityType == EIdentityType.Enterprise)
+				.WhereIF(IsCenter == false, (it, o) => o.ActualHandleOrgCode.StartsWith(_sessionContext.RequiredOrgId))
+				.GroupBy((it, o) => new { Id = it.Id.Substring(SqlFunc.MappingColumn<int>("0"), SqlFunc.MappingColumn<int>(count)) })
+				.Select((it, o) => new
+				{
+					HotspotCode = it.Id.Substring(SqlFunc.MappingColumn<int>("0"), SqlFunc.MappingColumn<int>(count)),
+					SumCount = SqlFunc.AggregateCount(it.HotSpotName)
+				})
+				.MergeTable()
+				.LeftJoin<Hotspot>((x, q) => x.HotspotCode == q.Id)
+				.Select((x, q) => new
+				{
+					SumCount = x.SumCount,
+					HotspotName = q.HotSpotName,
+				})
+				.ToDataTableAsync();
+
+			}
+            data.Columns["HotspotName"].SetOrdinal(0);
+			data.Columns["SumCount"].ColumnName = "分类统计";
+			data.Columns["HotspotName"].ColumnName = "热点名称";
+            //合计
+			DataRow sumRow = data.NewRow();
+			sumRow["热点名称"] = "合计";
+			decimal totalAmount = 0;
+			foreach (DataRow row in data.Rows)
+			{
+				totalAmount += Convert.ToDecimal(row["分类统计"]);
+			}
+			sumRow["分类统计"] = totalAmount;
+			data.Rows.Add(sumRow);
+
+			var stream = ExcelHelper.CreateStream(data);
+			return ExcelStreamResult(stream, "热点类型小类统计");
+		}
+
+		/// <summary>
+		/// 部门满意度统计
+		/// </summary>
+		/// <param name="dto"></param>
+		/// <returns></returns>
+		[HttpGet("visit-org-satisfaction-statistics")]
         public async Task<VisitAndOrgSatisfactionStatisticsResultDto> VisitAndOrgSatisfactionStatistics([FromQuery] PagedKeywordSonRequest dto)
         {
             var data = await _orderApplication.VisitAndOrgSatisfactionStatistics(dto);
@@ -2230,7 +2376,13 @@ namespace Hotline.Api.Controllers.Bi
 
             var queryData = await _orderReportApplication.GetDepartmentalProcessingStatisticsListNew(dto, HttpContext.RequestAborted)
                         .LeftJoin<Order>((x, o) => x.Id == o.Id)
-                        .OrderByDescending((x, o) => o.CreationTime)
+                        .OrderByIF(string.IsNullOrEmpty(dto.SortField), (x, o) => o.CreationTime, OrderByType.Desc)
+                        .OrderByIF(dto is { SortField: "startTime", SortRule: 0 }, (x, o) => o.StartTime, OrderByType.Asc) //受理时间升序
+                        .OrderByIF(dto is { SortField: "startTime", SortRule: 1 }, (x, o) => o.StartTime, OrderByType.Desc) //受理时间降序
+                        .OrderByIF(dto is { SortField: "expiredTime", SortRule: 0 }, (x, o) => o.ExpiredTime, OrderByType.Asc) //工单期满时间升序
+                        .OrderByIF(dto is { SortField: "expiredTime", SortRule: 1 }, (x, o) => o.ExpiredTime, OrderByType.Desc) //工单期满时间降序
+                        .OrderByIF(dto is { SortField: "filedTime", SortRule: 0 }, (x, o) => o.FiledTime, OrderByType.Asc)// 办结时间升序
+                        .OrderByIF(dto is { SortField: "filedTime", SortRule: 1 }, (x, o) => o.FiledTime, OrderByType.Desc) //办结时间降序
                          .Select((x, o) => new { o })
                         .ToPageListAsync(dto.PageIndex, dto.PageSize, total, HttpContext.RequestAborted);
 
@@ -2252,7 +2404,13 @@ namespace Hotline.Api.Controllers.Bi
         {
             var query = _orderReportApplication.GetDepartmentalProcessingStatisticsListNew(dto.QueryDto, HttpContext.RequestAborted)
                  .LeftJoin<Order>((x, o) => x.Id == o.Id)
-                 .OrderByDescending((x, o) => o.CreationTime)
+                .OrderByIF(string.IsNullOrEmpty(dto.QueryDto.SortField), (x, o) => o.CreationTime, OrderByType.Desc)
+                .OrderByIF(dto.QueryDto is { SortField: "startTime", SortRule: 0 }, (x, o) => o.StartTime, OrderByType.Asc) //受理时间升序
+                .OrderByIF(dto.QueryDto is { SortField: "startTime", SortRule: 1 }, (x, o) => o.StartTime, OrderByType.Desc) //受理时间降序
+                .OrderByIF(dto.QueryDto is { SortField: "expiredTime", SortRule: 0 }, (x, o) => o.ExpiredTime, OrderByType.Asc) //工单期满时间升序
+                .OrderByIF(dto.QueryDto is { SortField: "expiredTime", SortRule: 1 }, (x, o) => o.ExpiredTime, OrderByType.Desc) //工单期满时间降序
+                .OrderByIF(dto.QueryDto is { SortField: "filedTime", SortRule: 0 }, (x, o) => o.FiledTime, OrderByType.Asc)// 办结时间升序
+                .OrderByIF(dto.QueryDto is { SortField: "filedTime", SortRule: 1 }, (x, o) => o.FiledTime, OrderByType.Desc) //办结时间降序
                  .Select((x, o) => new { o });
             List<OrderDto> list = new List<OrderDto>();
             if (dto.IsExportAll)
@@ -2299,8 +2457,15 @@ namespace Hotline.Api.Controllers.Bi
 
             var queryData = await _orderReportApplication.DepartmentalProcessingStatisticsDetailsList(dto)
                         .LeftJoin<Order>((x, o) => x.Id == o.Id)
-                        .OrderByDescending((x, o) => o.CreationTime)
-                         .Select((x, o) => new { o })
+                        .OrderByIF(string.IsNullOrEmpty(dto.SortField),(x,o)=> o.CreationTime,OrderByType.Desc)
+                        .OrderByIF(dto is { SortField : "startTime", SortRule :0 },(x,o)=> o.StartTime,OrderByType.Asc) //受理时间升序
+                        .OrderByIF(dto is { SortField : "startTime", SortRule: 1 }, (x, o) => o.StartTime, OrderByType.Desc) //受理时间降序
+                        .OrderByIF(dto is { SortField: "expiredTime", SortRule:0 },(x,o)=> o.ExpiredTime,OrderByType.Asc) //工单期满时间升序
+                        .OrderByIF(dto is { SortField: "expiredTime", SortRule: 1 }, (x, o) => o.ExpiredTime, OrderByType.Desc) //工单期满时间降序
+                        .OrderByIF(dto is { SortField: "filedTime",SortRule:0 },(x,o)=> o.FiledTime,OrderByType.Asc)// 办结时间升序
+                        .OrderByIF(dto is { SortField: "filedTime",SortRule:1 },(x,o)=> o.FiledTime,OrderByType.Desc) //办结时间降序
+                        .Select((x, o) => new { o })
+                        
                         .ToPageListAsync(dto.PageIndex, dto.PageSize, total, HttpContext.RequestAborted);
 
             var dtos = queryData.Select(d =>
@@ -2321,7 +2486,13 @@ namespace Hotline.Api.Controllers.Bi
         {
             var query = _orderReportApplication.DepartmentalProcessingStatisticsDetailsList(dto.QueryDto)
                   .LeftJoin<Order>((x, o) => x.Id == o.Id)
-                  .OrderByDescending((x, o) => o.CreationTime)
+                  .OrderByIF(string.IsNullOrEmpty(dto.QueryDto.SortField), (x, o) => o.CreationTime, OrderByType.Desc)
+                        .OrderByIF(dto.QueryDto is { SortField: "startTime", SortRule: 0 }, (x, o) => o.StartTime, OrderByType.Asc) //受理时间升序
+                        .OrderByIF(dto.QueryDto is { SortField: "startTime", SortRule: 1 }, (x, o) => o.StartTime, OrderByType.Desc) //受理时间降序
+                        .OrderByIF(dto.QueryDto is { SortField: "expiredTime", SortRule: 0 }, (x, o) => o.ExpiredTime, OrderByType.Asc) //工单期满时间升序
+                        .OrderByIF(dto.QueryDto is { SortField: "expiredTime", SortRule: 1 }, (x, o) => o.ExpiredTime, OrderByType.Desc) //工单期满时间降序
+                        .OrderByIF(dto.QueryDto is { SortField: "filedTime", SortRule: 0 }, (x, o) => o.FiledTime, OrderByType.Asc)// 办结时间升序
+                        .OrderByIF(dto.QueryDto is { SortField: "filedTime", SortRule: 1 }, (x, o) => o.FiledTime, OrderByType.Desc) //办结时间降序
                   .Select((x, o) => new { o });
             var list = new List<OrderDto>();
             if (dto.IsExportAll)
@@ -3369,23 +3540,13 @@ namespace Hotline.Api.Controllers.Bi
         /// </summary>
         /// <returns></returns>
         [HttpGet("org-visitdetail-list-basedata")]
-        public async Task<object> OrgVisitDetailListBaseData()
+        public Dictionary<string, dynamic> OrgVisitDetailListBaseData()
         {
             return _baseDataApplication
                 .VisitSatisfaction()
                 .OrgsOptions(_sessionContext)
                 .AttitudeType()
                 .Build();
-            var VisitSatisfaction = _systemDicDataCacheManager.GetSysDicDataCache(SysDicTypeConsts.VisitSatisfaction);
-
-            if (_sessionContext.OrgIsCenter)
-            {
-                return new { OrgsOptions = await _systemOrganizeRepository.GetOrgJson(), VisitSatisfaction = VisitSatisfaction };
-            }
-            else
-            {
-                return new { OrgsOptions = await _systemOrganizeRepository.GetOrgJsonForUser(_sessionContext.RequiredOrgId), VisitSatisfaction = VisitSatisfaction };
-            }
         }
 
         /// <summary>

+ 6 - 18
src/Hotline.Api/Controllers/CommonPController.cs

@@ -228,13 +228,7 @@ namespace Hotline.Api.Controllers
 			{
 				//待办
 				var waitedDataList = await _orderRepository
-					.Queryable(hasHandled: false, isAdmin: isAdmin)
-					.Where(d => SqlFunc.Subqueryable<WorkflowStep>()
-						.Where(step => step.ExternalId == d.Id && step.Status != EWorkflowStepStatus.Handled &&
-						               ((step.FlowAssignType == EFlowAssignType.User && !string.IsNullOrEmpty(step.HandlerId) && step.HandlerId == _sessionContext.RequiredUserId) ||
-						                (step.FlowAssignType == EFlowAssignType.Org && !string.IsNullOrEmpty(step.HandlerOrgId) && step.HandlerOrgId == _sessionContext.RequiredOrgId) ||
-						                (step.FlowAssignType == EFlowAssignType.Role && !string.IsNullOrEmpty(step.RoleId) && _sessionContext.Roles.Contains(step.RoleId))))
-						.Any())
+					.Queryable(hasHandledStep: false, isAdmin: isAdmin)
 					.Includes(d => d.OrderSpecials)
 					.Where(d => d.Status != EOrderStatus.WaitForAccept && d.Status != EOrderStatus.BackToUnAccept && d.Status != EOrderStatus.SpecialToUnAccept && d.Status != EOrderStatus.HandOverToUnAccept)
 					.Where(d => d.Source < ESource.MLSQ || d.Source > ESource.WZSC)
@@ -298,7 +292,7 @@ namespace Hotline.Api.Controllers
 				//allNum += signDataList.Count;
 				//allList.AddRange(signDataList);
 				//延期待审批
-				var delayDataList = await _orderDelayRepository.Queryable(canView: true, isAdmin: isAdmin)
+				var delayDataList = await _orderDelayRepository.Queryable(hasHandledStep: false, isAdmin: isAdmin)
 					.Includes(d => d.Order)
 					.Where(d => d.DelayState == EDelayState.Examining)
 					.OrderByDescending(d => d.ApplyDelayTime)
@@ -344,7 +338,7 @@ namespace Hotline.Api.Controllers
 				//allNum += nearlyExpiredDataList.Count;
 				//allList.AddRange(nearlyExpiredDataList);
 				//甄别待审批
-				var screenDataList = await _orderScreenRepository.Queryable(hasHandled: !true, isAdmin: isAdmin)
+				var screenDataList = await _orderScreenRepository.Queryable(hasHandledStep: false, isAdmin: isAdmin)
 					.Includes(d => d.Order)
 					.Includes(d => d.ScreenDetails.Where(sd => sd.AuditUserId == _sessionContext.UserId).OrderByDescending(sd => sd.AuditTime).Take(1).ToList())
 					.Where(d => (d.Status == EScreenStatus.Apply || d.Status == EScreenStatus.Approval || (d.Status == EScreenStatus.SendBack && d.SendBackApply == false)))
@@ -410,13 +404,7 @@ namespace Hotline.Api.Controllers
 			{
 				//待办
 				var waitedDataList = await _orderRepository
-					.Queryable(hasHandled: false, isAdmin: isAdmin)
-					.Where(d => SqlFunc.Subqueryable<WorkflowStep>()
-						.Where(step => step.ExternalId == d.Id && step.Status != EWorkflowStepStatus.Handled &&
-						               ((step.FlowAssignType == EFlowAssignType.User && !string.IsNullOrEmpty(step.HandlerId) && step.HandlerId == _sessionContext.RequiredUserId) ||
-						                (step.FlowAssignType == EFlowAssignType.Org && !string.IsNullOrEmpty(step.HandlerOrgId) && step.HandlerOrgId == _sessionContext.RequiredOrgId) ||
-						                (step.FlowAssignType == EFlowAssignType.Role && !string.IsNullOrEmpty(step.RoleId) && _sessionContext.Roles.Contains(step.RoleId))))
-						.Any())
+					.Queryable(hasHandledStep: false, isAdmin: isAdmin)
 					.Includes(d => d.OrderSpecials)
 					.Where(d => d.Status != EOrderStatus.WaitForAccept && d.Status != EOrderStatus.BackToUnAccept && d.Status != EOrderStatus.SpecialToUnAccept && d.Status != EOrderStatus.HandOverToUnAccept)
 					.Where(d => d.Source < ESource.MLSQ || d.Source > ESource.WZSC)
@@ -460,7 +448,7 @@ namespace Hotline.Api.Controllers
 				//allNum += signDataList.Count;
 				//allList.AddRange(signDataList);
 				//甄别待审批
-				var screenDataList = await _orderScreenRepository.Queryable(hasHandled: !true, isAdmin: isAdmin)
+				var screenDataList = await _orderScreenRepository.Queryable(hasHandledStep: false, isAdmin: isAdmin)
 					.Includes(d => d.Order)
 					.Includes(d => d.ScreenDetails.Where(sd => sd.AuditUserId == _sessionContext.UserId).OrderByDescending(sd => sd.AuditTime).Take(1).ToList())
 					.Where(d => (d.Status == EScreenStatus.Apply || d.Status == EScreenStatus.Approval || (d.Status == EScreenStatus.SendBack && d.SendBackApply == false)))
@@ -480,7 +468,7 @@ namespace Hotline.Api.Controllers
 				//allNum += screenDataList.Count;
 				//allList.AddRange(screenDataList);
 				//延期待审批
-				var delayDataList = await _orderDelayRepository.Queryable(canView: true, isAdmin: isAdmin)
+				var delayDataList = await _orderDelayRepository.Queryable(hasHandledStep: false, isAdmin: isAdmin)
 					.Includes(d => d.Order)
 					.Where(d => d.DelayState == EDelayState.Examining)
 					.OrderByDescending(d => d.ApplyDelayTime)

+ 5 - 2
src/Hotline.Api/Controllers/ExportWordController.cs

@@ -46,7 +46,7 @@ namespace Hotline.Api.Controllers
         {
             var streams = new Dictionary<string, Stream>();
             var path = $"{Directory.GetCurrentDirectory()}/Template/AssignmentForm.doc";
-
+            int num = 1;
             foreach (var item in Ids)
             {
                 var order = await _orderRepository.GetAsync(item, HttpContext.RequestAborted);
@@ -95,7 +95,10 @@ namespace Hotline.Api.Controllers
                     exportTest.VisitOrg = visit;
                 }
                 if (Ids.Count > 1)
-                    streams.Add(order.No + path.Substring(path.LastIndexOf(".")), _wordHelperService.WordStream(path, exportTest));
+                {
+                    streams.Add(order.No + "_" + num + path.Substring(path.LastIndexOf(".")), _wordHelperService.WordStream(path, exportTest));
+                    num = num + 1;
+                }
                 else
                 {
                     var btyes = _wordHelperService.WordByte(path, exportTest);

+ 61 - 7
src/Hotline.Api/Controllers/KnowledgeController.cs

@@ -10,6 +10,7 @@ using Hotline.Application.Systems;
 using Hotline.Application.Tools;
 using Hotline.File;
 using Hotline.FlowEngine.WorkflowModules;
+using Hotline.FlowEngine.Workflows;
 using Hotline.KnowledgeBase;
 using Hotline.KnowledgeBase.Notifies;
 using Hotline.Permissions;
@@ -24,6 +25,7 @@ using Hotline.Share.Dtos.FlowEngine;
 using Hotline.Share.Dtos.Knowledge;
 using Hotline.Share.Dtos.Order;
 using Hotline.Share.Enums.Article;
+using Hotline.Share.Enums.FlowEngine;
 using Hotline.Share.Enums.KnowledgeBase;
 using Hotline.Share.Mq;
 using Hotline.Share.Tools;
@@ -36,6 +38,7 @@ using Microsoft.AspNetCore.Mvc;
 using Microsoft.EntityFrameworkCore.Metadata.Internal;
 using Org.BouncyCastle.Utilities.IO;
 using SqlSugar;
+using System.Text;
 using System.Threading;
 using XF.Domain.Authentications;
 using XF.Domain.Exceptions;
@@ -172,7 +175,7 @@ namespace Hotline.Api.Controllers
 
             kn.Status = EKnowledgeStatus.Drafts;
             kn.InitId();
-            if (dto.Data.Files.Any()) kn.FileJson = await _fileRepository.AddFileAsync(dto.Data.Files, kn.Id, "", HttpContext.RequestAborted);
+            if (dto.Data.Files.NotNullOrEmpty()) kn.FileJson = await _fileRepository.AddFileAsync(dto.Data.Files, kn.Id, "", HttpContext.RequestAborted);
             await _knowledgeRepository.AddAsync(kn, HttpContext.RequestAborted);
 
             if (dto.Data.KnowledgeType.Any())
@@ -323,15 +326,65 @@ namespace Hotline.Api.Controllers
 
         }
 
-
         /// <summary>
         /// 批量审核
         /// </summary>
         /// <returns></returns>
         [HttpPost("batch_audit")]
-        public async Task KnowledgeBatchAuditAsync()
+        public async Task<string> KnowledgeBatchAuditAsync([FromBody] KnowledgeBatchAuditInDto dto)
         {
-            // TODO: qcy 批量审核 
+            var result = new StringBuilder();
+            var fail = 0;
+            var success = 0;
+            var nextWorkflowDto = new NextWorkflowDto
+            {
+                IsSms = dto.IsSms,
+                ReviewResult = dto.IsPass ? EReviewResult.Approval : EReviewResult.Failed,
+                Opinion = dto.Opinion
+            };
+            foreach (var knowledgeId in dto.KnowledgeIds)
+            {
+                try
+                {
+                    var knowledge = await _knowledgeDomainService.KnowledgeInfo(knowledgeId, HttpContext.RequestAborted);
+                    nextWorkflowDto.WorkflowId = knowledge.WorkflowId;
+                    NextStepsWithOpinionDto<NextStepOption> next = null;
+                    try
+                    {
+                        next = await _workflowApplication.GetNextStepsAsync(knowledge.WorkflowId, HttpContext.RequestAborted);
+                    }
+                    catch (UserFriendlyException e)
+                    {
+                        if (e.Message.Contains("未找到对应节点"))
+                        {
+                            result.Append("无权审核:" + knowledge.Title);
+                            fail++;
+                        }
+                        else
+                        {
+                            throw;
+                        }
+                    }
+                    if (next == null) continue;
+                    nextWorkflowDto.StepId = next.StepId;
+                    nextWorkflowDto.NextStepCode = next.Steps.First().Key;
+                    nextWorkflowDto.NextStepName = next.Steps.First().Value;
+                    if (dto.IsPass)
+                        await _workflowApplication.NextAsync(nextWorkflowDto, cancellationToken: HttpContext.RequestAborted);
+                    else
+                    {
+                        var reject = nextWorkflowDto.Adapt<RejectDto>();
+                        await _workflowApplication.RejectAsync(reject, HttpContext.RequestAborted);
+                    }
+                    success++;
+                }
+                catch (UserFriendlyException e)
+                {
+                    result.Append(e.Message);
+                    fail++;
+                }
+            }
+            return $"总共: {dto.KnowledgeIds.Length}, 成功: {success}, 失败: {fail}, 失败原因: {result.ToString()}";
         }
 
         /// <summary>
@@ -586,7 +639,7 @@ namespace Hotline.Api.Controllers
         /// </summary>
         /// <returns></returns>
         [HttpGet("knowledge-status-data")]
-        public async Task<Dictionary<string, dynamic>> KnowledgeStatus()
+        public Dictionary<string, dynamic> KnowledgeStatus()
         {
             var tabNames = new List<KeyValuePair<int, string>>
             {
@@ -653,7 +706,7 @@ namespace Hotline.Api.Controllers
         /// </summary>
         /// <returns></returns>
         [HttpGet("knowretrieval/base_data")]
-        public async Task<Dictionary<string, dynamic>> GetKnowretrievalBaseData()
+        public Dictionary<string, dynamic> GetKnowretrievalBaseData()
         {
             return _baseDataApplication
                 .KnowledgeRetrievalType(new[]{3, 4})
@@ -679,6 +732,7 @@ namespace Hotline.Api.Controllers
                 .ToPagedListAsync(pagedDto.PageIndex, pagedDto.PageSize);
             return new PagedDto<KnowledgeWorkFlowDto>(total, _mapper.Map<IReadOnlyList<KnowledgeWorkFlowDto>>(temp));
         }
+
         /// <summary>
         /// 知识查重
         /// </summary>
@@ -1475,7 +1529,7 @@ namespace Hotline.Api.Controllers
         /// <param name="dto"></param>
         /// <returns></returns>
         [HttpGet("hotword/basedata")]
-        public async Task<Dictionary<string, dynamic>> AddKnowledgeHotWordBaseDataAsync()
+        public Dictionary<string, dynamic> AddKnowledgeHotWordBaseDataAsync()
         {
             return _baseDataApplication
                 .KnowledgeHotWordType()

+ 308 - 385
src/Hotline.Api/Controllers/OrderController.cs

@@ -66,6 +66,9 @@ using XF.Utility.EnumExtensions;
 using Hotline.Application.Contracts.Validators.FlowEngine;
 using Hotline.Authentications;
 using Microsoft.AspNetCore.Components;
+using Quartz.Simpl;
+using static Lucene.Net.Util.Fst.Util;
+using DocumentFormat.OpenXml.Spreadsheet;
 
 namespace Hotline.Api.Controllers;
 
@@ -305,8 +308,8 @@ public class OrderController : BaseController
             .WhereIF(!string.IsNullOrEmpty(dto.AcceptType), d => d.AcceptTypeCode == dto.AcceptType) //受理类型
             .WhereIF(!string.IsNullOrEmpty(dto.Hotspot), d => d.HotspotSpliceName != null && d.HotspotSpliceName.Contains(dto.Hotspot))
             .WhereIF(!string.IsNullOrEmpty(dto.FromPhone), d => d.FromPhone == dto.FromPhone) //来电号码
-            //.WhereIF(!string.IsNullOrEmpty(dto.PubMan),
-            //    d => d.AcceptorName.Contains(dto.PubMan!) || d.AcceptorStaffNo.Contains(dto.PubMan!))
+                                                                                              //.WhereIF(!string.IsNullOrEmpty(dto.PubMan),
+                                                                                              //    d => d.AcceptorName.Contains(dto.PubMan!) || d.AcceptorStaffNo.Contains(dto.PubMan!))
             .WhereIF(dto.PubRange == EPublicState.Pub, d => d.OrderPublish.PublishState)
             .WhereIF(dto.PubRange == EPublicState.NoPub, d => !d.OrderPublish.PublishState)
             .WhereIF(dto.IsProvinceOrder.HasValue && dto.IsProvinceOrder == true, d => d.Source == ESource.ProvinceStraight)
@@ -358,132 +361,7 @@ public class OrderController : BaseController
             {
                 try
                 {
-                    OrderPublish orderPublish = new OrderPublish();
-                    orderPublish.OrderId = order.Id;
-                    orderPublish.No = order.No;
-                    orderPublish.PublishState = false; //当前写死为false
-                    orderPublish.ArrangeTitle = order.Title;
-                    orderPublish.ArrangeContent = order.Content;
-                    orderPublish.ArrangeOpinion = order.FileOpinion;
-                    orderPublish.ProPublishState = false;
-                    orderPublish.FeedBackPhone = order.Contact;
-                    orderPublish.CreatorName = _sessionContext.UserName;
-                    await _orderPublishRepository.AddAsync(orderPublish);
-                    order.Publish(orderPublish.PublishState);
-                    await _orderRepository.UpdateAsync(order);
-                    //推省上
-                    var publishPublishOrder = _mapper.Map<PublishPublishOrderDto>(orderPublish);
-                    publishPublishOrder.Order = _mapper.Map<OrderDto>(order);
-                    await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderPublishOrder, publishPublishOrder);
-
-                    //推应急管理局
-                    //是否开启
-                    var isOpenContingencyManagement =
-                        _systemSettingCacheManager.GetSetting(SettingConstants.IsOpenContingencyManagement)?.SettingValue[0];
-                    if (isOpenContingencyManagement == "true")
-                        await _publisher.PublishAsync(new ContingencyManagementNotify(order, order.Title, order.Content, order.ActualOpinion),
-                            PublishStrategy.ParallelWhenAll, HttpContext.RequestAborted);
-
-                    var orderVisit = new OrderVisit();
-                    orderVisit.No = order.No;
-                    orderVisit.OrderId = order.Id;
-                    orderVisit.VisitState = EVisitState.WaitForVisit;
-                    orderVisit.PublishTime = DateTime.Now;
-                    orderVisit.IsCanHandle = true;
-                    orderVisit.EmployeeId = _sessionContext.RequiredUserId;
-
-                    if (_appOptions.Value.IsZiGong)
-                    {
-                        orderVisit.EmployeeId = string.Empty;
-                    }
-
-                    if (order is { ProcessType: EProcessType.Zhiban, CounterSignType: null } && !order.IsProvince)
-                    {
-                        orderVisit.VisitState = EVisitState.Visited;
-                        orderVisit.VisitTime = DateTime.Now;
-                        orderVisit.VisitType = EVisitType.OtherVisit;
-                        orderVisit.NowEvaluate = new Kv() { Key = "4", Value = "满意" };
-                        if (_appOptions.Value.IsZiGong)
-                        {
-                            // 根据禅道 自贡需求 Id_361, 第一条, 3小条需求;
-                            // 直办件归档后自动回访量需统计在“胡玲”的默认回访量中;
-                            orderVisit.EmployeeId = _systemSettingCacheManager.DefaultVisitEmployeeId;
-                        }
-                    }
-
-                    if (order.CounterSignType != ECounterSignType.Center)
-                    {
-                        orderVisit.IsCanAiVisit = true;
-                    }
-
-
-                    string visitId = await _orderVisitRepository.AddAsync(orderVisit);
-
-                    //新增回访信息
-                    var visitedDetail = new List<OrderVisitDetail>();
-
-                    var seatDetail = new OrderVisitDetail();
-                    seatDetail.VisitId = visitId;
-                    seatDetail.VisitTarget = EVisitTarget.Seat;
-
-
-                    var orgDetail = new OrderVisitDetail();
-                    orgDetail.VisitId = visitId;
-                    orgDetail.VisitOrgCode = order.ActualHandleOrgCode;
-                    orgDetail.VisitOrgName = order.ActualHandleOrgName;
-                    orgDetail.VisitTarget = EVisitTarget.Org;
-                    if (order is { ProcessType: EProcessType.Zhiban, CounterSignType: null, IsProvince: false })
-                    {
-                        var satisfy = new Kv() { Key = "4", Value = "满意" };
-                        orgDetail.OrgProcessingResults = satisfy;
-                        //orgDetail.OrgHandledAttitude = satisfy;
-                    }
-
-                    visitedDetail.Add(orgDetail);
-
-                    if (order is { ProcessType: EProcessType.Zhiban, CounterSignType: null })
-                    {
-                        seatDetail.VoiceEvaluate = EVoiceEvaluate.Satisfied;
-                        seatDetail.SeatEvaluate = ESeatEvaluate.Satisfied;
-                        order.Visited("4", "满意");
-                        order.Status = EOrderStatus.Visited;
-                        await _orderRepository.UpdateAsync(order, HttpContext.RequestAborted);
-                    }
-
-                    visitedDetail.Add(seatDetail);
-                    await _orderVisitedDetailRepository.AddRangeAsync(visitedDetail, HttpContext.RequestAborted);
-
-                    if (orderVisit.VisitState == EVisitState.Visited)
-                    {
-                        //推省上
-                        await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderVisited,
-                            new PublishVisitDto()
-                            {
-                                Order = _mapper.Map<OrderDto>(order),
-                                No = orderVisit.No,
-                                VisitType = orderVisit.VisitType,
-                                VisitName = orderVisit.CreatorName,
-                                VisitTime = orderVisit.VisitTime,
-                                VisitRemark = orderVisit.NowEvaluate?.Value,
-                                AreaCode = order.AreaCode!,
-                                SubjectResultSatifyCode = orderVisit.NowEvaluate?.Key,
-                                FirstSatisfactionCode = orderVisit.NowEvaluate?.Key,
-                                ClientGuid = ""
-                            }, cancellationToken: HttpContext.RequestAborted);
-                    }
-
-                    //推门户
-                    await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderVisitedWeb, new PublishVisitAllDto()
-                    {
-                        Id = orderVisit.Id,
-                        Order = _mapper.Map<OrderDto>(order),
-                        OrderVisitDetails = _mapper.Map<List<VisitDetailDto>>(orderVisit.OrderVisitDetails),
-                        VisitName = _sessionContext.UserName,
-                        VisitTime = orderVisit.VisitTime,
-                        VisitType = orderVisit.VisitType,
-                        VisitState = orderVisit.VisitState,
-                        PublishTime = orderVisit.PublishTime,
-                    }, cancellationToken: HttpContext.RequestAborted);
+                    await _orderDomainService.OrderPublishAsync(order, HttpContext.RequestAborted);
                 }
                 catch
                 {
@@ -888,6 +766,7 @@ public class OrderController : BaseController
             .WhereIF(dto.VisitStateQuery == EVisitStateQuery.Visited, d => d.VisitState == EVisitState.Visited)
             .WhereIF(dto.VisitStateQuery == EVisitStateQuery.SMSUnsatisfied, d => d.VisitState == EVisitState.SMSUnsatisfied)
             .WhereIF(dto.VisitStateQuery == EVisitStateQuery.SMSVisiting, d => d.VisitState == EVisitState.SMSVisiting)
+            .WhereIF(dto.VisitStateQuery == EVisitStateQuery.NoPutThrough , d=>d.IsPutThrough == false)
             .WhereIF(!string.IsNullOrEmpty(dto.Keyword), d => d.Order.Title.StartsWith(dto.Keyword!))
             .WhereIF(!string.IsNullOrEmpty(dto.No), d => d.No == dto.No)
             .WhereIF(dto.VisitType != null, d => d.VisitType == dto.VisitType)
@@ -936,7 +815,7 @@ public class OrderController : BaseController
     /// </summary>
     /// <returns></returns>
     [HttpGet("visit/basedata")]
-    public async Task<object> VisitBaseData()
+    public Dictionary<string, dynamic> VisitBaseData()
         => _baseDataApplication
             .VisitType()
             .VoiceEvaluate()
@@ -1181,7 +1060,7 @@ public class OrderController : BaseController
     /// <param name="dto"></param>
     /// <returns></returns>
     [HttpGet("visit/batch-basedata")]
-    public async Task<dynamic> VisitBatchBaseDat()
+    public Dictionary<string, dynamic> VisitBatchBaseDat()
         => _baseDataApplication
             .SeatEvaluate()
             .DissatisfiedReason()
@@ -1318,16 +1197,29 @@ public class OrderController : BaseController
         }).ToList();
     }
 
-    #endregion
-
-    #region 二次回访申请
 
     /// <summary>
-    /// 可二次回访申请列表
+    /// 设置未接通
     /// </summary>
     /// <param name="dto"></param>
     /// <returns></returns>
-    [HttpGet("visitapply/visitagainlist")]
+    [HttpPut("visit/put_through")]
+	public async Task VisitPutThrough([FromBody] VisitPutThroughDto dto ) {
+
+        await _orderVisitRepository.Updateable().SetColumns(x => new OrderVisit { IsPutThrough = false }).Where(x => x.Id == dto.id).ExecuteCommandAsync(HttpContext.RequestAborted);
+    }
+
+
+	#endregion
+
+	#region 二次回访申请
+
+	/// <summary>
+	/// 可二次回访申请列表
+	/// </summary>
+	/// <param name="dto"></param>
+	/// <returns></returns>
+	[HttpGet("visitapply/visitagainlist")]
     public async Task<PagedDto<OrderCanVisitAgainDto>> OrderVisitAgainList([FromQuery] OrderVisitAgainListDto dto)
     {
         var (total, items) = await _orderVisitedDetailRepository.Queryable()
@@ -1756,23 +1648,34 @@ public class OrderController : BaseController
     [HttpGet("delay/{workflowId}/nextsteps")]
     public async Task<NextStepsDto> OrderDelayNextsteps(string workflowId)
     {
-        var workflow = await _workflowRepository.GetAsync(workflowId, HttpContext.RequestAborted);
+        //var workflow = await _workflowRepository.GetAsync(workflowId, HttpContext.RequestAborted);
+        var workflow = await _workflowDomainService.GetWorkflowAsync(workflowId, withDefine: true, withSteps: true,
+	        cancellationToken: HttpContext.RequestAborted);
+        var currentStep = workflow.Steps.FirstOrDefault(d => d.Status == EWorkflowStepStatus.WaitForAccept || d.Status == EWorkflowStepStatus.WaitForHandle);
         if (workflow != null)
         {
-            var orderDelay = await _orderDelayRepository.Queryable().Includes(x => x.Order).Where(x => x.Id == workflow.ExternalId)
+	    
+			var orderDelay = await _orderDelayRepository.Queryable().Includes(x => x.Order).Where(x => x.Id == workflow.ExternalId)
                 .FirstAsync(HttpContext.RequestAborted);
             if (orderDelay != null)
             {
-                var result = await _workflowApplication.GetNextStepsAsync(workflowId, HttpContext.RequestAborted);
-                if (!orderDelay.Order.IsProvince)
+				var result = await _workflowApplication.GetNextStepsAsync(workflowId, HttpContext.RequestAborted);
+				if (!orderDelay.Order.IsProvince)
                 {
                     if (result.Steps.Any(x => x.Value == "省审批"))
                     {
                         result.Steps.Remove(result.Steps.First(x => x.Value == "省审批"));
                     }
                 }
+                if (!_sessionContext.OrgIsCenter && currentStep.Name != "中心初审")
+                {
+	                if (result.Steps.Any(x => x.Value == "中心终审"))
+	                {
+		                result.Steps.Remove(result.Steps.First(x => x.Value == "中心终审"));
+	                }
+                }
 
-                return result;
+				return result;
             }
         }
 
@@ -1838,7 +1741,7 @@ public class OrderController : BaseController
             if (!_sessionContext.OrgIsCenter)
             {
                 query.Where(d => d.AutomaticDelayNum == 0 || d.AutomaticDelayNum == null);
-			}
+            }
         }
 
         var (total, items) = await query
@@ -1938,9 +1841,23 @@ public class OrderController : BaseController
     [HttpGet("delay/startflow")]
     public async Task<NextStepsDto> GetDelayFlowStartOptions()
     {
-        return await _workflowApplication.GetStartStepsAsync(WorkflowModuleConsts.OrderDelay,
+        var result = await _workflowApplication.GetStartStepsAsync(WorkflowModuleConsts.OrderDelay,
             HttpContext.RequestAborted);
-    }
+        if (_sessionContext.OrgIsCenter)
+        {
+            if (result.Steps.Any(x => x.Value == "中心初审"))
+            {
+                result.Steps.Remove(result.Steps.First(x => x.Value == "中心初审"));
+            }
+        }
+        else {
+	        if (result.Steps.Any(x => x.Value == "中心终审"))
+	        {
+		        result.Steps.Remove(result.Steps.First(x => x.Value == "中心终审"));
+	        }
+		}
+        return result;
+	}
 
     /// <summary>
     /// 延期页面基础信息
@@ -1979,103 +1896,61 @@ public class OrderController : BaseController
             // dto.CreationTimeStart = _timeLimitDomainService.CalcWorkTimeReduce(DateTime.Now, 5);
             dto.CreationTimeStart = await _expireTime.CalcWorkTimeReduce(DateTime.Now, 5);
         }
-
-        var query = _orderVisitedDetailRepository.Queryable(false, true)
-            .Includes(x => x.OrderVisit)
-            .Includes(x => x.OrderVisit, y => y.Order)
-            .Includes(x => x.OrderVisit, y => y.Employee)
-            //.LeftJoin<OrderScreen>((x, s) => x.Id == s.VisitDetailId && s.IsDeleted == false)
-            .Includes(x => x.OrderScreens)
-            .Where(x => x.OrderScreens.Any(s => s.Status == EScreenStatus.SendBack && s.ScreenType == dto.ScreenType && s.SendBackApply == true) ||
-                        x.OrderScreens.Any() == false
-                //|| x.OrderScreens.Any(s => (s.Status != EScreenStatus.SendBack && s.SendBackApply != true)) == false
-            )
-            .WhereIF(dto.ScreenType == EOrderScreenType.Seat, x => x.OrderVisit.Order.IsProvince == false)
-            .WhereIF(dto.ScreenSendBack is 1,
-                x => x.OrderScreens.Any(s => s.Status == EScreenStatus.SendBack && s.ScreenType == dto.ScreenType && s.SendBackApply == true))
-            .WhereIF(dto.ScreenSendBack is 2,
-                x => x.OrderScreens.Any(s => (s.Status != EScreenStatus.SendBack && s.ScreenType == dto.ScreenType && s.SendBackApply != true)) ==
-                     false)
-            .WhereIF(!string.IsNullOrEmpty(dto.No), x => x.OrderVisit.Order!.No!.Contains(dto.No!))
-            .WhereIF(!string.IsNullOrEmpty(dto.Title), x => x.OrderVisit.Order!.Title!.Contains(dto.Title!))
-            .WhereIF(dto.IsProvince.HasValue, x => x.OrderVisit.Order!.IsProvince == dto.IsProvince)
-            .WhereIF(!string.IsNullOrEmpty(dto.AcceptType), x => x.OrderVisit.Order!.AcceptTypeCode! == dto.AcceptType!)
-            .WhereIF(!string.IsNullOrEmpty(dto.HotspotSpliceName),
-                x => x.OrderVisit.Order!.Hotspot.HotSpotFullName!.StartsWith(dto.HotspotSpliceName!))
-            .WhereIF(!string.IsNullOrEmpty(dto.SourceChannel), x => x.OrderVisit.Order!.SourceChannelCode! == dto.SourceChannel!)
-            .WhereIF(!string.IsNullOrEmpty(dto.OrgLevelOneName), x => x.OrderVisit.Order!.OrgLevelOneName!.Contains(dto.OrgLevelOneName!))
-            .WhereIF(!string.IsNullOrEmpty(dto.CurrentHandleOrgName),
-                x => x.OrderVisit.Order!.CurrentHandleOrgName!.Contains(dto.CurrentHandleOrgName!))
-            .WhereIF(!string.IsNullOrEmpty(dto.VisitOrgName), x => x.VisitOrgName!.Contains(dto.VisitOrgName!))
-            .WhereIF(dto.CreationTime.HasValue && dto.EndCreationTime.HasValue,
-                x => x.OrderVisit.Order!.CreationTime >= dto.CreationTime && x.OrderVisit.Order!.CreationTime <= dto.EndCreationTime)
-            .WhereIF(dto.CurrentHandleTime.HasValue && dto.EndCurrentHandleTime.HasValue,
-                x => x.OrderVisit.Order!.CurrentHandleTime >= dto.CurrentHandleTime &&
-                     x.OrderVisit.Order!.ActualHandleTime <= dto.EndCurrentHandleTime)
-            .WhereIF(dto.FiledTime.HasValue && dto.EndFiledTime.HasValue,
-                x => x.OrderVisit.Order!.FiledTime >= dto.FiledTime && x.OrderVisit.Order!.FiledTime <= dto.EndFiledTime)
-            .WhereIF(dto.VisitTime.HasValue && dto.EndVisitTime.HasValue,
-                x => x.OrderVisit.VisitTime >= dto.VisitTime && x.OrderVisit.VisitTime <= dto.EndVisitTime)
-            .WhereIF(dto.IsHomePage.HasValue && dto.IsHomePage == true,
-                x => x.OrderVisit.VisitTime < dto.CreationTimeEnd && x.OrderVisit.VisitTime > dto.CreationTimeStart)
-            //.WhereIF(dto.CounterSignType.HasValue, x => x.OrderVisit.Order!.CounterSignType == dto.CounterSignType)
-            //.WhereIF(!string.IsNullOrEmpty(dto.OrgProcessingResults),
-            //    x => SqlFunc.JsonField(x.OrgProcessingResults, "Key") == dto.OrgProcessingResults)
-            //.WhereIF(!string.IsNullOrEmpty(dto.OrgHandledAttitude),
-            //    x => SqlFunc.JsonListObjectAny(x.OrgHandledAttitude, "Key", dto.OrgHandledAttitude))
-            //.WhereIF(!string.IsNullOrEmpty(dto.OrgNoSatisfiedReason),
-            //    x => SqlFunc.JsonField(x.OrgNoSatisfiedReason, "Key") == dto.OrgNoSatisfiedReason)
-            .Where(x => x.OrderVisit.VisitState == EVisitState.Visited && x.OrderVisit.IsCanHandle);
-        if (_sessionContext.OrgId != null && !_sessionContext.OrgIsCenter)
-        {
-            query.WhereIF(!string.IsNullOrEmpty(dto.Keyword),
-                    x => x.OrderVisit.Order.Title.Contains(dto.Keyword!) ||
-                         x.OrderVisit.Order.No.Contains(dto.Keyword!))
-                .Where(x => x.VisitTarget == EVisitTarget.Org && x.VisitOrgCode == _sessionContext.OrgId && (
-                    SqlFunc.JsonField(x.OrgProcessingResults, "Key") == "1" ||
-                    SqlFunc.JsonField(x.OrgProcessingResults, "Key") == "2" ||
-                    SqlFunc.JsonField(x.OrgHandledAttitude, "Key") == "1" ||
-                    SqlFunc.JsonField(x.OrgHandledAttitude, "Key") == "2"
-                ));
-        }
-        else
-        {
-            query.WhereIF(!string.IsNullOrEmpty(dto.Keyword),
-                    x => x.OrderVisit.Order.Title.Contains(dto.Keyword!) ||
-                         x.OrderVisit.Order.No.Contains(dto.Keyword!))
-                .WhereIF(dto.ScreenType == EOrderScreenType.Org, x => x.VisitTarget == EVisitTarget.Org && (
-                    SqlFunc.JsonField(x.OrgProcessingResults, "Key") == "1" ||
-                    SqlFunc.JsonField(x.OrgProcessingResults, "Key") == "2" ||
-                    SqlFunc.JsonField(x.OrgHandledAttitude, "Key") == "1" ||
-                    SqlFunc.JsonField(x.OrgHandledAttitude, "Key") == "2"
-                ))
-                .WhereIF(dto.ScreenType == EOrderScreenType.Seat,
-                    x => x.VisitTarget == EVisitTarget.Seat &&
-                         (x.SeatEvaluate == ESeatEvaluate.VeryNoSatisfied || x.SeatEvaluate == ESeatEvaluate.NoSatisfied))
-                ;
-        }
-
-        var (total, items) = await query
-            .OrderByIF(dto is { SortRule: 0, SortField: "order.startTime" }, x => x.OrderVisit.Order.StartTime, OrderByType.Asc)
-            .OrderByIF(dto is { SortRule: 1, SortField: "order.startTime" }, x => x.OrderVisit.Order.StartTime, OrderByType.Desc)
-            .OrderByIF(dto is { SortRule: 0, SortField: "order.actualHandleTime" }, x => x.OrderVisit.Order.ActualHandleTime, OrderByType.Asc)
-            .OrderByIF(dto is { SortRule: 1, SortField: "order.actualHandleTime" }, x => x.OrderVisit.Order.ActualHandleTime, OrderByType.Desc)
-            .OrderByIF(dto is { SortRule: 0, SortField: "order.filedTime" }, x => x.OrderVisit.Order.FiledTime, OrderByType.Asc)
-            .OrderByIF(dto is { SortRule: 1, SortField: "order.filedTime" }, x => x.OrderVisit.Order.FiledTime, OrderByType.Desc)
-            .OrderByIF(dto is { SortRule: 0, SortField: "orderVisit.visitTime" }, x => x.OrderVisit.VisitTime, OrderByType.Asc)
-            .OrderByIF(dto is { SortRule: 1, SortField: "orderVisit.visitTime" }, x => x.OrderVisit.VisitTime, OrderByType.Desc)
-            .OrderByIF(dto.SortRule is null, x => x.CreationTime, OrderByType.Desc)
-            .ToPagedListAsync(dto.PageIndex, dto.PageSize, HttpContext.RequestAborted);
+		var (total, items) = await _orderApplication.MayScreenList(dto)
+			.ToPagedListAsync(dto.PageIndex, dto.PageSize, HttpContext.RequestAborted);
         return new PagedDto<OrderVisitDetailDto>(total, _mapper.Map<IReadOnlyList<OrderVisitDetailDto>>(items));
     }
 
-
     /// <summary>
-    /// 工单甄别列表
+    /// 工单甄别待申请列表导出
     /// </summary>
-    /// <param name="dto"></param>
     /// <returns></returns>
-    [HttpGet("screen")]
+    [HttpPost("mayscreen/_export")]
+    public async Task<FileStreamResult> ScreenListExport([FromBody] ExportExcelDto<MayScreenListDto> dto)
+    {
+	    if (_appOptions.Value.IsYiBin) dto.QueryDto.ScreenType = EOrderScreenType.Org;
+
+	    dto.QueryDto.CreationTimeEnd = DateTime.Now;
+	    dto.QueryDto.CreationTimeStart = DateTime.Now;
+	    if (dto.QueryDto.IsHomePage != null && dto.QueryDto.IsHomePage == true)
+	    {
+		    // dto.CreationTimeStart = _timeLimitDomainService.CalcWorkTimeReduce(DateTime.Now, 5);
+		    dto.QueryDto.CreationTimeStart = await _expireTime.CalcWorkTimeReduce(DateTime.Now, 5);
+	    }
+
+		var query = _orderApplication.MayScreenList(dto.QueryDto);
+	    List<OrderVisitDetail> data;
+	    if (dto.IsExportAll)
+	    {
+		    data = await query.ToListAsync(HttpContext.RequestAborted);
+	    }
+	    else
+	    {
+		    var (_, items) = await query.ToPagedListAsync(dto.QueryDto, HttpContext.RequestAborted);
+		    data = items;
+	    }
+
+	    var dataDtos = _mapper.Map<ICollection<OrderVisitDetailDto>>(data);
+
+	    dynamic? dynamicClass = DynamicClassHelper.CreateDynamicClass(dto.ColumnInfos);
+
+	    var dtos = dataDtos
+		    .Select(stu => _mapper.Map(stu, typeof(OrderVisitDetailDto), dynamicClass))
+		    .Cast<object>()
+		    .ToList();
+
+	    var stream = ExcelHelper.CreateStream(dtos);
+
+	    return ExcelStreamResult(stream, "工单甄别待申请列表数据");
+    }
+
+
+	/// <summary>
+	/// 工单甄别列表
+	/// </summary>
+	/// <param name="dto"></param>
+	/// <returns></returns>
+	[HttpGet("screen")]
     public async Task<PagedDto<OrderScreenListDto>> ScreenList([FromQuery] ScreenListDto dto)
     {
         var (total, items) = await _orderApplication.OrderScreenList(dto)
@@ -2264,9 +2139,22 @@ public class OrderController : BaseController
     {
         //return await _workflowApplication.GetStartOptionsAsync(WorkflowModuleConsts.OrderScreen,
         //    HttpContext.RequestAborted);
-
-        return await _workflowApplication.GetStartStepsAsync(WorkflowModuleConsts.OrderScreen,
+        var result = await _workflowApplication.GetStartStepsAsync(WorkflowModuleConsts.OrderScreen,
             HttpContext.RequestAborted);
+        if (_sessionContext.OrgIsCenter)
+        {
+            if (result.Steps.Any(x => x.Value == "中心初审"))
+            {
+                result.Steps.Remove(result.Steps.First(x => x.Value == "中心初审"));
+            }
+        }
+        else {
+			if (result.Steps.Any(x => x.Value == "中心班长"))
+			{
+				result.Steps.Remove(result.Steps.First(x => x.Value == "中心班长"));
+			}
+		}
+		return result;
     }
 
     /// <summary>
@@ -2304,6 +2192,13 @@ public class OrderController : BaseController
                     }
                 }
 
+                if (!_sessionContext.OrgIsCenter)
+                {
+					if (result.Steps.Any(x => x.Value == "中心班长"))
+					{
+						result.Steps.Remove(result.Steps.First(x => x.Value == "中心班长"));
+					}
+				}
                 return result;
             }
         }
@@ -3151,7 +3046,7 @@ public class OrderController : BaseController
                 cancellationToken: HttpContext.RequestAborted);
 
             List<OrderRemarksDto> remarks = workflow.Steps.Where(x => !string.IsNullOrEmpty(x.Remark)).Select(x => new OrderRemarksDto
-                { Remark = x.Remark, RemarkTime = x.CreationTime, RemarkUser = x.CreatorName }).ToList();
+            { Remark = x.Remark, RemarkTime = x.HandleTime, RemarkUser = x.HandlerName }).ToList();
             dto.OrderRemarks = remarks;
             if (order.Status == EOrderStatus.SendBack || order.Status == EOrderStatus.SendBackAudit || order.Status == EOrderStatus.BackToUnAccept)
             {
@@ -3589,7 +3484,7 @@ public class OrderController : BaseController
             var startDto = _mapper.Map<StartWorkflowDto>(dto.Workflow);
             startDto.DefinitionModuleCode = WorkflowModuleConsts.OrderHandle;
             startDto.Title = order.Title;
-            var (workflow,startStep) = await _workflowDomainService.StartAsync(startDto, order.Id, order.ExpiredTime, HttpContext.RequestAborted);
+            var (workflow, startStep) = await _workflowDomainService.StartAsync(startDto, order.Id, order.ExpiredTime, HttpContext.RequestAborted);
 
             await HandleOrderAsync(order, workflow, startStep, dto.Data, dto.Workflow, HttpContext.RequestAborted);
         }
@@ -3599,7 +3494,7 @@ public class OrderController : BaseController
             throw new UserFriendlyException($"工单开启流程失败!, {e.Message}, {e.StackTrace}", "工单开启流程失败");
         }
 
-        if (_appOptions.Value.IsYiBin && dto.Data.Transpond.HasValue && dto.Data.Transpond.Value)
+        if (_appOptions.Value.IsYiBin && order.Transpond.HasValue && order.Transpond.Value)
         {
             //开启流程处理事件,处理市州互转
             await _publisher.PublishAsync(new OrderStartWorkflowNotify(order.Id), PublishStrategy.ParallelWhenAll,
@@ -3656,19 +3551,20 @@ public class OrderController : BaseController
     }
 
     private async Task HandleOrderAsync(Order order, Workflow workflow, WorkflowStep startStep, OrderHandleFlowDto orderHandleFlowDto,
-        BasicWorkflowDto workflowDto,
-        CancellationToken cancellationToken)
+        BasicWorkflowDto workflowDto, CancellationToken cancellationToken)
     {
+        var isAutoFillSummaryOpinion = bool.Parse(_systemSettingCacheManager.GetSetting(SettingConstants.IsAutoFillSummaryOpinion).SettingValue[0]);
+        
         switch (orderHandleFlowDto.OrderAssignMode)
         {
             case EOrderAssignMode.AdjoinLevel:
                 var nextDto = _mapper.Map<NextWorkflowDto>(workflowDto);
                 nextDto.WorkflowId = startStep.WorkflowId;
                 nextDto.StepId = startStep.Id;
-
-                // 宜宾需求: 1.是否是判断节点  2.是否存在历史派单节点  3.存在获取上个派单节点  4.不存在走平均派单
+                
                 if (workflowDto.BusinessType == EBusinessType.Send)
                 {
+                    // 宜宾需求: 1.是否是派单节点  2.是否存在历史派单节点  3.存在获取上个派单节点  4.不存在走平均派单
                     if (_appOptions.Value.IsYiBin)
                     {
                         var sendOrderTraces = workflow.Traces.Where(x => x.BusinessType == EBusinessType.Send);
@@ -3689,34 +3585,16 @@ public class OrderController : BaseController
                         }
                         else
                         {
-                            // 平均派单
-                            var averageSendOrder = bool.Parse(_systemSettingCacheManager.GetSetting(SettingConstants.AverageSendOrder).SettingValue[0]);
-                            if (averageSendOrder)
-                            {
-                                if (!nextDto.NextHandlers.Any())
-                                {
-                                    var handler = await _orderDomainService.AverageOrder(HttpContext.RequestAborted);
-                                    nextDto.NextHandlers = new List<FlowStepHandler> { handler };
-                                }
-                            }
+                            await AverageSendOrderAsync(nextDto, HttpContext.RequestAborted);
                         }
                     }
                     else
                     {
-                        // 平均派单
-                        var averageSendOrder = bool.Parse(_systemSettingCacheManager.GetSetting(SettingConstants.AverageSendOrder).SettingValue[0]);
-                        if (averageSendOrder)
-                        {
-                            if (!nextDto.NextHandlers.Any())
-                            {
-                                var handler = await _orderDomainService.AverageOrder(cancellationToken);
-                                nextDto.NextHandlers = new List<FlowStepHandler> { handler };
-                            }
-                        }
+                        await AverageSendOrderAsync(nextDto, HttpContext.RequestAborted);
                     }
                 }
 
-                await _workflowDomainService.NextAsync(_sessionContext, nextDto, order.ExpiredTime, cancellationToken);
+                await _workflowDomainService.NextAsync(_sessionContext, nextDto, order.ExpiredTime,isAutoFillSummaryOpinion, cancellationToken);
                 break;
             case EOrderAssignMode.CrossLevel:
                 if (!orderHandleFlowDto.CrossSteps.Any())
@@ -3746,7 +3624,8 @@ public class OrderController : BaseController
                             OrgId = unhandleStep.HandlerOrgId,
                             OrgName = unhandleStep.HandlerOrgName,
                         };
-                        var nextSteps = await _workflowDomainService.NextAsync(operater, nextflowDto, order.ExpiredTime, cancellationToken);
+                        var nextSteps = await _workflowDomainService.NextAsync(operater, nextflowDto, order.ExpiredTime,
+                            isAutoFillSummaryOpinion, cancellationToken);
                         tempSteps.AddRange(nextSteps);
                     }
 
@@ -3761,6 +3640,20 @@ public class OrderController : BaseController
         }
     }
 
+    private async Task AverageSendOrderAsync(NextWorkflowDto nextDto, CancellationToken cancellationToken)
+    {
+        // 平均派单
+        var averageSendOrder = bool.Parse(_systemSettingCacheManager.GetSetting(SettingConstants.AverageSendOrder).SettingValue[0]);
+        if (averageSendOrder)
+        {
+            if (!nextDto.NextHandlers.Any())
+            {
+                var handler = await _orderDomainService.AverageOrder(cancellationToken);
+                nextDto.NextHandlers = new List<FlowStepHandler> { handler };
+            }
+        }
+    }
+
     /// <summary>
     /// 跨级指派查询下一步可选节点及办理对象参数
     /// </summary>
@@ -4018,6 +3911,22 @@ public class OrderController : BaseController
         await _orderDomainService.TriggerAverageOrder(HttpContext.RequestAborted);
     }
 
+    /// <summary>
+    /// 批量归档
+    /// </summary>
+    [HttpPost("batch-file")]
+    public async Task BatchFile([FromBody]OrderBatchFileDto dto)
+    {
+        var orders = await _orderRepository.Queryable()
+            .Where(d => dto.OrderIds.Contains(d.Id))
+            .ToListAsync(HttpContext.RequestAborted);
+        foreach (var order in orders)
+        {
+            await _workflowDomainService.JumpToEndAsync(_sessionContext, order.WorkflowId,dto.Opinion,dto.Files,
+                order.ExpiredTime, cancellationToken: HttpContext.RequestAborted);
+        }
+    }
+
     #endregion
 
     #region 工单待办
@@ -4029,10 +3938,10 @@ public class OrderController : BaseController
     public async Task<PagedDto<OrderDto>> QueryWaited([FromQuery] QueryOrderWaitedDto dto)
     {
         var isHandledStep = dto.IsHandled.HasValue && dto.IsHandled.Value;
-        if (!isHandledStep)
-	        dto.QueryType = null;
+        if (isHandledStep)
+            dto.QueryType = null;
 
-		var isAdmin = _orderDomainService.IsCheckAdmin();
+        var isAdmin = _orderDomainService.IsCheckAdmin();
         var query = _orderRepository
             .Queryable(hasHandledStep: isHandledStep, isAdmin: isAdmin)
             .Includes(d => d.OrderSpecials);
@@ -4070,7 +3979,14 @@ public class OrderController : BaseController
             .WhereIF(dto.EndTime.HasValue, d => d.StartTime <= dto.EndTime)
             .WhereIF(dto.IsUrgent.HasValue, d => d.IsUrgent == dto.IsUrgent!.Value)
             .WhereIF(dto.Status.HasValue, d => d.Status == dto.Status)
-            .OrderByDescending(d => new { d.IsUrgent, d.StartTime })
+            .OrderByDescending(d=> d.IsUrgent)
+            .OrderByIF(string.IsNullOrEmpty(dto.SortField),d => d.StartTime ,OrderByType.Desc)
+            .OrderByIF(dto is { SortField: "creationTime", SortRule:0 },d=> d.CreationTime,OrderByType.Asc) //创建时间升序
+            .OrderByIF(dto is { SortField: "creationTime", SortRule: 1 }, d => d.CreationTime, OrderByType.Desc) //创建时间降序
+            .OrderByIF(dto is { SortField: "startTime", SortRule: 0 }, d => d.StartTime, OrderByType.Asc) //受理时间升序
+            .OrderByIF(dto is { SortField: "startTime", SortRule: 1 }, d => d.StartTime, OrderByType.Desc) //受理时间降序
+            .OrderByIF(dto is { SortField: "expiredTime", SortRule: 0 }, d => d.ExpiredTime, OrderByType.Asc) //期满时间升序
+            .OrderByIF(dto is { SortField: "expiredTime", SortRule: 1 }, d => d.ExpiredTime, OrderByType.Desc) //期满时间降序
             .ToPagedListAsync(dto, HttpContext.RequestAborted);
 
         return new PagedDto<OrderDto>(total, _mapper.Map<IReadOnlyList<OrderDto>>(items));
@@ -4160,7 +4076,11 @@ public class OrderController : BaseController
             .WhereIF(!string.IsNullOrEmpty(dto.CenterToOrgHandlerName), d => d.CenterToOrgHandlerName == dto.CenterToOrgHandlerName)
             .WhereIF(dto.IsUrgent.HasValue, d => d.IsUrgent == dto.IsUrgent!.Value)
             .OrderBy(d => d.Status)
-            .OrderBy(d => d.CreationTime, OrderByType.Desc)
+            .OrderByIF(string.IsNullOrEmpty(dto.SortField),d => d.CreationTime, OrderByType.Desc)
+            .OrderByIF(dto is { SortField: "startTime", SortRule: 0 }, d => d.StartTime, OrderByType.Asc) //受理时间升序
+            .OrderByIF(dto is { SortField: "startTime", SortRule: 1 }, d => d.StartTime, OrderByType.Desc) //受理时间降序
+            .OrderByIF(dto is { SortField: "expiredTime", SortRule: 0 }, d => d.ExpiredTime, OrderByType.Asc) //期满时间升序
+            .OrderByIF(dto is { SortField: "expiredTime", SortRule: 1 }, d => d.ExpiredTime, OrderByType.Desc) //期满时间降序
             .ToPagedListAsync(dto, HttpContext.RequestAborted);
 
         return new PagedDto<OrderDto>(total, _mapper.Map<IReadOnlyList<OrderDto>>(items));
@@ -4877,27 +4797,28 @@ public class OrderController : BaseController
 
             if (dto.FlowDirection is EFlowDirection.OrgToOrg)
             {
-	            expiredTime.ExpiredTime = order.ExpiredTime.Value;
-	            expiredTime.NearlyExpiredTime = order.NearlyExpiredTime.Value;
-	            expiredTime.NearlyExpiredTimeOne = order.NearlyExpiredTimeOne.Value;
-			}
-            else {
-				if (_appOptions.Value.IsZiGong)
-				{
-					if (dto.FlowDirection != null)
-					{
-						expiredTime = await _expireTime.CalcExpiredTime(DateTime.Now, dto.FlowDirection.Value, order.Adapt<OrderTimeClacInfo>());
-					}
-					else
-					{
-						expiredTime = await _expireTime.CalcEndTime(DateTime.Now, order.AcceptTypeCode);
-					}
-				}
-				else
-				{
-					expiredTime = await _expireTime.CalcEndTime(DateTime.Now, order.AcceptTypeCode);
-				}
-			}
+                expiredTime.ExpiredTime = order.ExpiredTime.Value;
+                expiredTime.NearlyExpiredTime = order.NearlyExpiredTime.Value;
+                expiredTime.NearlyExpiredTimeOne = order.NearlyExpiredTimeOne.Value;
+            }
+            else
+            {
+                if (_appOptions.Value.IsZiGong)
+                {
+                    if (dto.FlowDirection != null)
+                    {
+                        expiredTime = await _expireTime.CalcExpiredTime(DateTime.Now, dto.FlowDirection.Value, order.Adapt<OrderTimeClacInfo>());
+                    }
+                    else
+                    {
+                        expiredTime = await _expireTime.CalcEndTime(DateTime.Now, order.AcceptTypeCode);
+                    }
+                }
+                else
+                {
+                    expiredTime = await _expireTime.CalcEndTime(DateTime.Now, order.AcceptTypeCode);
+                }
+            }
 
             var processType = dto.FlowDirection is EFlowDirection.OrgToCenter or EFlowDirection.CenterToCenter or EFlowDirection.FiledToCenter
                 ? EProcessType.Zhiban
@@ -4913,13 +4834,13 @@ public class OrderController : BaseController
             //	ETimeType.WorkDay,
             //	dto.TimeLimit.Value, order.AcceptTypeCode);
             await _orderRepository.Updateable().SetColumns(o => new Orders.Order()
-                {
-                    ExpiredTime = expiredTime.ExpiredTime,
-                    NearlyExpiredTime = expiredTime.NearlyExpiredTime,
-                    NearlyExpiredTimeOne = expiredTime.NearlyExpiredTimeOne,
-                    ProcessType = processType,
-                    Status = EOrderStatus.Special
-                })
+            {
+                ExpiredTime = expiredTime.ExpiredTime,
+                NearlyExpiredTime = expiredTime.NearlyExpiredTime,
+                NearlyExpiredTimeOne = expiredTime.NearlyExpiredTimeOne,
+                ProcessType = processType,
+                Status = EOrderStatus.Special
+            })
                 .Where(o => o.Id == order.Id).ExecuteCommandAsync(HttpContext.RequestAborted);
             var orderDto = _mapper.Map<OrderDto>(order);
             await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderExpiredTimeUpdate, orderDto,
@@ -5075,11 +4996,11 @@ public class OrderController : BaseController
 
                 endTime = expiredTime.EndTime;
                 await _orderRepository.Updateable().SetColumns(o => new Orders.Order()
-                    {
-                        ExpiredTime = expiredTime.EndTime,
-                        NearlyExpiredTime = expiredTime.NearlyExpiredTime,
-                        NearlyExpiredTimeOne = expiredTime.NearlyExpiredTimeOne
-                    })
+                {
+                    ExpiredTime = expiredTime.EndTime,
+                    NearlyExpiredTime = expiredTime.NearlyExpiredTime,
+                    NearlyExpiredTimeOne = expiredTime.NearlyExpiredTimeOne
+                })
                     .Where(o => o.Id == order.Id).ExecuteCommandAsync(HttpContext.RequestAborted);
                 var orderDto = _mapper.Map<OrderDto>(order);
                 await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderExpiredTimeUpdate, orderDto,
@@ -5171,50 +5092,50 @@ public class OrderController : BaseController
                 HandlerType = special.HandlerType.Value,
                 BusinessType = special.BusinessType.Value
             };
-			ExpiredTimeWithConfig? expiredTime = new ExpiredTimeWithConfig();
+            ExpiredTimeWithConfig? expiredTime = new ExpiredTimeWithConfig();
 
-			if (dto.FlowDirection is EFlowDirection.OrgToOrg)
-			{
-				expiredTime.ExpiredTime = order.ExpiredTime.Value;
-				expiredTime.NearlyExpiredTime = order.NearlyExpiredTime.Value;
-				expiredTime.NearlyExpiredTimeOne = order.NearlyExpiredTimeOne.Value;
-			}
-			else
-			{
-				if (_appOptions.Value.IsZiGong)
-				{
-					if (dto.FlowDirection != null)
-					{
-						expiredTime = await _expireTime.CalcExpiredTime(DateTime.Now, dto.FlowDirection.Value, order.Adapt<OrderTimeClacInfo>());
-					}
-					else
-					{
-						expiredTime = await _expireTime.CalcEndTime(DateTime.Now, order.AcceptTypeCode);
-					}
-				}
-				else
-				{
-					expiredTime = await _expireTime.CalcEndTime(DateTime.Now, order.AcceptTypeCode);
-				}
-			}
-			if (expiredTime.ExpiredTime < order.ExpiredTime)
-			{
-				expiredTime.ExpiredTime = order.ExpiredTime.Value;
-				expiredTime.NearlyExpiredTime = order.NearlyExpiredTime.Value;
-				expiredTime.NearlyExpiredTimeOne = order.NearlyExpiredTimeOne.Value;
-			}
+            if (dto.FlowDirection is EFlowDirection.OrgToOrg)
+            {
+                expiredTime.ExpiredTime = order.ExpiredTime.Value;
+                expiredTime.NearlyExpiredTime = order.NearlyExpiredTime.Value;
+                expiredTime.NearlyExpiredTimeOne = order.NearlyExpiredTimeOne.Value;
+            }
+            else
+            {
+                if (_appOptions.Value.IsZiGong)
+                {
+                    if (dto.FlowDirection != null)
+                    {
+                        expiredTime = await _expireTime.CalcExpiredTime(DateTime.Now, dto.FlowDirection.Value, order.Adapt<OrderTimeClacInfo>());
+                    }
+                    else
+                    {
+                        expiredTime = await _expireTime.CalcEndTime(DateTime.Now, order.AcceptTypeCode);
+                    }
+                }
+                else
+                {
+                    expiredTime = await _expireTime.CalcEndTime(DateTime.Now, order.AcceptTypeCode);
+                }
+            }
+            if (expiredTime.ExpiredTime < order.ExpiredTime)
+            {
+                expiredTime.ExpiredTime = order.ExpiredTime.Value;
+                expiredTime.NearlyExpiredTime = order.NearlyExpiredTime.Value;
+                expiredTime.NearlyExpiredTimeOne = order.NearlyExpiredTimeOne.Value;
+            }
 
-			var processType = special.FlowDirection is EFlowDirection.OrgToCenter or EFlowDirection.CenterToCenter or EFlowDirection.FiledToCenter
+            var processType = special.FlowDirection is EFlowDirection.OrgToCenter or EFlowDirection.CenterToCenter or EFlowDirection.FiledToCenter
                 ? EProcessType.Zhiban
                 : EProcessType.Jiaoban;
             await _orderRepository.Updateable().SetColumns(o => new Orders.Order()
-                {
-                    ExpiredTime = expiredTime.ExpiredTime,
-                    NearlyExpiredTime = expiredTime.NearlyExpiredTime,
-                    NearlyExpiredTimeOne = expiredTime.NearlyExpiredTimeOne,
-                    ProcessType = processType,
-                    Status = EOrderStatus.Special
-                })
+            {
+                ExpiredTime = expiredTime.ExpiredTime,
+                NearlyExpiredTime = expiredTime.NearlyExpiredTime,
+                NearlyExpiredTimeOne = expiredTime.NearlyExpiredTimeOne,
+                ProcessType = processType,
+                Status = EOrderStatus.Special
+            })
                 .Where(o => o.Id == order.Id).ExecuteCommandAsync(HttpContext.RequestAborted);
             var orderDto = _mapper.Map<OrderDto>(order);
             await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderExpiredTimeUpdate, orderDto,
@@ -5306,49 +5227,49 @@ public class OrderController : BaseController
                     HandlerType = special.HandlerType.Value,
                     BusinessType = special.BusinessType.Value
                 };
-				ExpiredTimeWithConfig? expiredTime = new ExpiredTimeWithConfig();
+                ExpiredTimeWithConfig? expiredTime = new ExpiredTimeWithConfig();
 
-				if (dto.FlowDirection is EFlowDirection.OrgToOrg)
-				{
-					expiredTime.ExpiredTime = order.ExpiredTime.Value;
-					expiredTime.NearlyExpiredTime = order.NearlyExpiredTime.Value;
-					expiredTime.NearlyExpiredTimeOne = order.NearlyExpiredTimeOne.Value;
-				}
-				else
-				{
-					if (_appOptions.Value.IsZiGong)
-					{
-						if (dto.FlowDirection != null)
-						{
-							expiredTime = await _expireTime.CalcExpiredTime(DateTime.Now, dto.FlowDirection.Value, order.Adapt<OrderTimeClacInfo>());
-						}
-						else
-						{
-							expiredTime = await _expireTime.CalcEndTime(DateTime.Now, order.AcceptTypeCode);
-						}
-					}
-					else
-					{
-						expiredTime = await _expireTime.CalcEndTime(DateTime.Now, order.AcceptTypeCode);
-					}
-				}
-				if (expiredTime.ExpiredTime < order.ExpiredTime)
-				{
-					expiredTime.ExpiredTime = order.ExpiredTime.Value;
-					expiredTime.NearlyExpiredTime = order.NearlyExpiredTime.Value;
-					expiredTime.NearlyExpiredTimeOne = order.NearlyExpiredTimeOne.Value;
-				}
-				var processType = special.FlowDirection is EFlowDirection.OrgToCenter or EFlowDirection.CenterToCenter or EFlowDirection.FiledToCenter
+                if (dto.FlowDirection is EFlowDirection.OrgToOrg)
+                {
+                    expiredTime.ExpiredTime = order.ExpiredTime.Value;
+                    expiredTime.NearlyExpiredTime = order.NearlyExpiredTime.Value;
+                    expiredTime.NearlyExpiredTimeOne = order.NearlyExpiredTimeOne.Value;
+                }
+                else
+                {
+                    if (_appOptions.Value.IsZiGong)
+                    {
+                        if (dto.FlowDirection != null)
+                        {
+                            expiredTime = await _expireTime.CalcExpiredTime(DateTime.Now, dto.FlowDirection.Value, order.Adapt<OrderTimeClacInfo>());
+                        }
+                        else
+                        {
+                            expiredTime = await _expireTime.CalcEndTime(DateTime.Now, order.AcceptTypeCode);
+                        }
+                    }
+                    else
+                    {
+                        expiredTime = await _expireTime.CalcEndTime(DateTime.Now, order.AcceptTypeCode);
+                    }
+                }
+                if (expiredTime.ExpiredTime < order.ExpiredTime)
+                {
+                    expiredTime.ExpiredTime = order.ExpiredTime.Value;
+                    expiredTime.NearlyExpiredTime = order.NearlyExpiredTime.Value;
+                    expiredTime.NearlyExpiredTimeOne = order.NearlyExpiredTimeOne.Value;
+                }
+                var processType = special.FlowDirection is EFlowDirection.OrgToCenter or EFlowDirection.CenterToCenter or EFlowDirection.FiledToCenter
                     ? EProcessType.Zhiban
                     : EProcessType.Jiaoban;
                 await _orderRepository.Updateable().SetColumns(o => new Orders.Order()
-                    {
-                        ExpiredTime = expiredTime.ExpiredTime,
-                        NearlyExpiredTime = expiredTime.NearlyExpiredTime,
-                        NearlyExpiredTimeOne = expiredTime.NearlyExpiredTimeOne,
-                        ProcessType = processType,
-                        Status = EOrderStatus.Special
-                    })
+                {
+                    ExpiredTime = expiredTime.ExpiredTime,
+                    NearlyExpiredTime = expiredTime.NearlyExpiredTime,
+                    NearlyExpiredTimeOne = expiredTime.NearlyExpiredTimeOne,
+                    ProcessType = processType,
+                    Status = EOrderStatus.Special
+                })
                     .Where(o => o.Id == order.Id).ExecuteCommandAsync(HttpContext.RequestAborted);
                 var orderDto = _mapper.Map<OrderDto>(order);
                 await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderExpiredTimeUpdate, orderDto,
@@ -5491,7 +5412,7 @@ public class OrderController : BaseController
                 d => d.Title.Contains(dto.Keyword!) || d.No.Contains(dto.Keyword!))
             //.WhereIF(!string.IsNullOrEmpty(dto.Content), d => d.Content.Contains(dto.Content!))
             .WhereIF(!string.IsNullOrEmpty(dto.AcceptType), d => d.AcceptTypeCode == dto.AcceptType) //受理类型
-            //.WhereIF(dto.AcceptTypes.Any(), d => dto.AcceptTypes.Contains(d.AcceptTypeCode)) //受理类型
+                                                                                                     //.WhereIF(dto.AcceptTypes.Any(), d => dto.AcceptTypes.Contains(d.AcceptTypeCode)) //受理类型
             .WhereIF(!string.IsNullOrEmpty(dto.Channel), d => d.SourceChannelCode == dto.Channel) //来源渠道
             .WhereIF(!string.IsNullOrEmpty(dto.Hotspot), d => d.HotspotSpliceName != null && d.HotspotSpliceName.Contains(dto.Hotspot))
             .WhereIF(!string.IsNullOrEmpty(dto.TransferPhone), d => d.TransferPhone.Contains(dto.TransferPhone!))
@@ -6286,7 +6207,8 @@ public class OrderController : BaseController
     [HttpGet("order/about_expire/list")]
     public async Task<PagedDto<OrderDto>> AboutList([FromQuery] AboutToExpireListDto dto)
     {
-        var (total, items) = await _orderApplication.GetAboutToExpireAsync(dto).ToPagedListAsync(dto, HttpContext.RequestAborted);
+        var (total, items) = await _orderApplication.GetAboutToExpireAsync(dto)
+            .ToPagedListAsync(dto, HttpContext.RequestAborted);
         return new PagedDto<OrderDto>(total, _mapper.Map<IReadOnlyList<OrderDto>>(items));
     }
 
@@ -6338,7 +6260,8 @@ public class OrderController : BaseController
     [HttpGet("order/expire/list")]
     public async Task<PagedDto<OrderDto>> ExpireList([FromQuery] AboutToExpireListDto dto)
     {
-        var (total, items) = await _orderApplication.GetToExpireAsync(dto).ToPagedListAsync(dto, HttpContext.RequestAborted);
+        var (total, items) = await _orderApplication.GetToExpireAsync(dto)
+            .ToPagedListAsync(dto, HttpContext.RequestAborted);
         return new PagedDto<OrderDto>(total, _mapper.Map<IReadOnlyList<OrderDto>>(items));
     }
 
@@ -7228,7 +7151,7 @@ public class OrderController : BaseController
         {
             await _orderRepository.Updateable()
                 .SetColumns(o => new Orders.Order()
-                    { SignerId = dto.Handler.UserId, SignerName = dto.Handler.Username, Status = EOrderStatus.HandOverToUnAccept })
+                { SignerId = dto.Handler.UserId, SignerName = dto.Handler.Username, Status = EOrderStatus.HandOverToUnAccept })
                 .Where(o => o.Id == dto.OrderId).ExecuteCommandAsync(HttpContext.RequestAborted);
         }
         else

+ 1 - 1
src/Hotline.Api/Controllers/OrderRevocationController.cs

@@ -156,7 +156,7 @@ namespace Hotline.Api.Controllers
                             }
                             else
                             {
-                                await _workflowApplication.JumpToEndAsync(_sessionContext, order.WorkflowId, dto.RevocationReason,
+                                await _workflowDomainService.JumpToEndAsync(_sessionContext, order.WorkflowId, dto.RevocationReason,
                                     null, order.ExpiredTime, cancellationToken: HttpContext.RequestAborted);
                             }
 

+ 1 - 1
src/Hotline.Api/config/appsettings.Development.json

@@ -68,7 +68,7 @@
     "Host": "110.188.24.182",
     "Port": 50179,
     "Password": "fengwo123!$!$",
-    "Database": 3 //release:3, dev:5
+    "Database": 3 //test:3, dev:5
   },
   "Swagger": true,
   "AccLog":  false,

+ 13 - 0
src/Hotline.Application.Contracts/Validators/Order/OrderBatchFileDtoValidator.cs

@@ -0,0 +1,13 @@
+using FluentValidation;
+using Hotline.Share.Dtos.Order.Handle;
+
+namespace Hotline.Application.Contracts.Validators.Order;
+
+public class OrderBatchFileDtoValidator:AbstractValidator<OrderBatchFileDto>
+{
+    public OrderBatchFileDtoValidator()
+    {
+        RuleFor(d => d.OrderIds).NotEmpty();
+        RuleFor(d=>d.Opinion).NotEmpty();
+    }
+}

+ 16 - 3
src/Hotline.Application.Tests/Application/KnowApplicationTest.cs

@@ -76,15 +76,14 @@ public class KnowApplicationTest
     [Fact]
     public async Task UpdateKnowledgeWord_Test()
     {
-
         var entity = await _knowledgeHotWordRepository.Queryable()
             .OrderByDescending(m => m.CreationTime)
             .FirstAsync();
         entity.KeyWord = "单元测试修改";
         var inDto = entity.Adapt<UpdateKnowledgeHotWordInDto>();
         await _knowApplication.UpdateKnowledgeHotWordAsync(inDto);
-        var updateEntity = await _knowledgeWordRepository.GetAsync(entity.Id);
-        updateEntity.Tag.ShouldBe(entity.KeyWord);
+        var updateEntity = await _knowledgeHotWordRepository.GetAsync(entity.Id);
+        updateEntity.KeyWord.ShouldBe(entity.KeyWord);
     }
 
     [Fact]
@@ -106,4 +105,18 @@ public class KnowApplicationTest
             e.Message.ShouldBe("热词已存在");
         }
     }
+
+    /// <summary>
+    /// 测试扩展方法是否能正确判断集合空和非空的情况
+    /// </summary>
+    [Fact]
+    public void ListEx_Test()
+    {
+        List<string> a = null;
+        a.IsNullOrEmpty().ShouldBeTrue();
+        a.NotNullOrEmpty().ShouldBeFalse();
+        a = new List<string>();
+        a.IsNullOrEmpty().ShouldBeTrue();
+        a.NotNullOrEmpty().ShouldBeFalse();
+    }
 }

+ 3 - 56
src/Hotline.Application.Tests/Application/OrderApplicationTest.cs

@@ -36,15 +36,8 @@ public class OrderApplicationTest
         _messageRepository = messageRepository;
     }
 
-    [Theory]
-    [InlineData(1)]
-    [InlineData(2)]
-    [InlineData(3)]
-    [InlineData(4)]
-    [InlineData(5)]
-    [InlineData(6)]
-    [InlineData(7)]
-    public async Task VisitPushSMS_Test(int count)
+    //[Fact]
+    public async Task VisitPushSMS_Test()
     {
         var orderVisit = await _orderVisitRepository.Queryable()
             .Where(m => m.VisitState == EVisitState.WaitForVisit)
@@ -61,53 +54,7 @@ public class OrderApplicationTest
         visit.VisitType.ShouldBe(EVisitType.SmsVisit);
     }
 
-    [Theory]
-    [InlineData("08dcd937-5800-4e44-81d7-68a318dbc251", "沟通地点", "张三", "13666666666", "63344B7C-D2CB-4B40-8B13-009923393573")]
-    public async Task SaveOrderWorkflowInfo_Test(string workflowId,
-        string realCommunicationAddress,
-        string realHandlerName,
-        string realHandlerPhone,
-        string transpondCityId
-        )
-    {
-        await _orderRepository.Updateable()
-            .SetColumns(m => m.RealCommunicationAddress == null)
-            .Where(m => m.WorkflowId == workflowId)
-            .ExecuteCommandAsync();
-        var time = DateTime.Now;
-        var dicSystem = _systemDicDataCacheManager.GetSysDicDataCache(SysDicTypeConsts.TranspondCity).First();
-        var dto = new NextWorkflowDto<OrderHandleFlowDto>
-        {
-            // RealCommunicationAddress = realCommunicationAddress,
-            // WorkflowId = workflowId,
-            // RealHandlerName = realHandlerName,
-            // RealHandlerPhone = realHandlerPhone,
-            // RealCommunicationMode = ERealCommunicationMode.Locale,
-            // RealCommunicationTime = time,
-            // RealIsContacted = true,
-            // RealContactLocale = true,
-            // IsOther = true,
-            // OtherRemark = "备注",
-            // TranspondCityId = dicSystem.Id,
-            // TranspondCityName = dicSystem.DicDataName,
-            // TranspondCityValue = dicSystem.DicDataValue,
-        };
-        var order = await _orderApplication.SaveOrderWorkflowInfo(dto, new CancellationToken());
-        order = await _orderRepository.GetAsync(order.Id);
-        order.RealCommunicationAddress.ShouldBe(realCommunicationAddress);
-        order.RealHandlerPhone.ShouldBe(realHandlerPhone);
-        order.RealHandlerName.ShouldBe(realHandlerName);
-        order.RealCommunicationMode.ShouldBe(ERealCommunicationMode.Locale);
-        order.RealCommunicationTime.Value.ToString("yyyy-MM-dd hh:mm:ss").ShouldBe(time.ToString("yyyy-MM-dd hh:mm:ss"));
-        order.RealIsContacted.ShouldBe(true);
-        order.RealContactLocale.ShouldBe(true);
-        order.IsOther.ShouldBe(true);
-        order.OtherRemark.ShouldBe("备注");
-        order.TranspondCityId.ShouldBe(dicSystem.Id);
-        order.TranspondCityName.ShouldBe(dicSystem.DicDataName);
-        order.TranspondCityValue.ShouldBe(dicSystem.DicDataValue);
-    }
-
+   
     [Fact]
     public async Task MapConfig_Test()
     {

+ 5 - 4
src/Hotline.Application.Tests/Application/ZiGongCallReportApplicationTest.cs

@@ -28,8 +28,8 @@ public class ZiGongCallReportApplicationTest
     {
         var inDto = new BiQueryCallsDto
         {
-            StartTime = "2024-07-29".ObjToDate(),
-            EndTime = "2024-07-29 23:59:59".ObjToDate()
+            StartTime = "2024-10-30".ObjToDate(),
+            EndTime = "2024-10-30 23:59:59".ObjToDate()
         };
         var (total, items) = await _ziGongCallReportApplication.QueryCallsDetailInTotalAsync(inDto, false);
         total.ShouldNotBe(0);
@@ -56,10 +56,11 @@ public class ZiGongCallReportApplicationTest
     [InlineData(null, null, null, "From")]
     public async Task QueryCallsStatisticsDetail_Test(string? orderNo, string? fromNo, string? toNo, string? endBy)
     {
+        return;
         var inDto = new QueryCallsStatisticsDetailInDto
         {
-            StartTime = "2024-07-29".ObjToDate(),
-            EndTime = "2024-07-29 23:59:59".ObjToDate(),
+            StartTime = "2024-10-30".ObjToDate(),
+            EndTime = "2024-10-30 23:59:59".ObjToDate(),
             OrderNo = orderNo,
             FromNo = fromNo,
             ToNo = toNo,

+ 10 - 5
src/Hotline.Application.Tests/Controller/DefaultHttpContextAccessor.cs

@@ -36,15 +36,20 @@ public class DefaultHttpContextAccessor : ISessionContext, IScopeDependency
     /// <summary>
     /// Id of current tenant or null for host
     /// </summary>
-    public string? UserId { get {
+    public string? UserId
+    {
+        get
+        {
             return TestSessionConstants.UserId;
-        } init { } }
+        }
+        init { }
+    }
 
     /// <summary>
     /// Id of current user or throw Exception for guest
     /// </summary>
     /// <exception cref="AuthenticationException"></exception>
-    public string RequiredUserId { get; }
+    public string RequiredUserId { get { return TestSessionConstants.UserId; } }
     public string? UserName { get { return TestSessionConstants.UserName; } init { } }
     public string? Phone { get; init; }
 
@@ -52,8 +57,8 @@ public class DefaultHttpContextAccessor : ISessionContext, IScopeDependency
     /// Roles
     /// </summary>
     public string[] Roles { get { return TestSessionConstants.Roles; } init { } }
-    public string? OrgId { get { return TestSessionConstants.OrgId;  } init { } }
-    public string RequiredOrgId { get; }
+    public string? OrgId { get { return TestSessionConstants.OrgId; } init { } }
+    public string RequiredOrgId { get { return TestSessionConstants.OrgId; } }
     public string? OrgName { get; init; }
     public int OrgLevel { get; init; }
     public string? OrgAreaCode { get; init; }

+ 70 - 0
src/Hotline.Application.Tests/Controller/KnowledgeControllerTest.cs

@@ -0,0 +1,70 @@
+using Hotline.Api.Controllers;
+using Hotline.Api.Controllers.Bi;
+using Hotline.Application.Tests.Mock;
+using Hotline.Identity.Accounts;
+using Hotline.Identity.Roles;
+using Hotline.KnowledgeBase;
+using Hotline.Share.Dtos.FlowEngine;
+using Hotline.Share.Dtos.Knowledge;
+using Hotline.Share.Enums.FlowEngine;
+using Hotline.Share.Enums.KnowledgeBase;
+using Hotline.Users;
+using Microsoft.AspNetCore.Http;
+using Microsoft.AspNetCore.Mvc;
+using Microsoft.Extensions.DependencyInjection;
+using Shouldly;
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+using XF.Domain.Repository;
+
+namespace Hotline.Application.Tests.Controller;
+public class KnowledgeControllerTest : TestBase
+{
+    private readonly KnowledgeServiceMock _knowledgeServiceMock;
+    private readonly KnowledgeController _knowledgeController;
+    private readonly IRepository<KnowledgeBase.Knowledge> _knowledgeRepository;
+
+    public KnowledgeControllerTest(IAccountRepository accountRepository, IRepository<Role> roleRepository, UserController userController, IServiceScopeFactory scopeFactory, IRepository<User> userRepository, KnowledgeServiceMock knowledgeServiceMock, KnowledgeController knowledgeController, IRepository<KnowledgeBase.Knowledge> knowledgeRepository) : base(accountRepository, roleRepository, userController, scopeFactory, userRepository)
+    {
+        _knowledgeServiceMock = knowledgeServiceMock;
+        _knowledgeController = knowledgeController;
+        _knowledgeController.ControllerContext = new ControllerContext
+        {
+            HttpContext = new DefaultHttpContext()
+        };
+        _knowledgeRepository = knowledgeRepository;
+    }
+
+    /// <summary>
+    /// 批量审核知识
+    /// </summary>
+    /// <returns></returns>
+    [Fact]
+    public async Task KnowledgeBatchAudit_TestAsync()
+    {
+        SetPaiDanYuan();
+        var inDto = new KnowledgeBatchAuditInDto
+        {
+            IsPass = true,
+            KnowledgeIds = new string[3],
+            IsSms = false,
+            Opinion = "批量审核通过"
+        };
+        for (int i = 0;i < 3;i++)
+        {
+            inDto.KnowledgeIds[i] = _knowledgeServiceMock.创建并审核知识()
+                .GetKnowledgeId();
+
+        }
+        Set班长();
+        var result = await _knowledgeController.KnowledgeBatchAuditAsync(inDto);
+        foreach (var id in inDto.KnowledgeIds)
+        {
+            var k = await _knowledgeRepository.GetAsync(id);
+            k.Status.ShouldBe(EKnowledgeStatus.OnShelf, $"{id} 状态错误: {result}");
+        }
+    }
+}

+ 108 - 20
src/Hotline.Application.Tests/Controller/OrderControllerTest.cs

@@ -1,16 +1,23 @@
 using AutoFixture;
+using Castle.DynamicProxy;
 using Hotline.Api.Controllers;
 using Hotline.Application.Tests.Infrastructure;
+using Hotline.Application.Tests.Mock;
+using Hotline.FlowEngine.Notifications;
+using Hotline.FlowEngine.WorkflowModules;
 using Hotline.Identity.Accounts;
 using Hotline.Identity.Roles;
 using Hotline.Orders;
 using Hotline.Settings.Hotspots;
 using Hotline.Share.Dtos.File;
+using Hotline.Share.Dtos.FlowEngine;
 using Hotline.Share.Dtos.Order;
 using Hotline.Share.Dtos.Users;
+using Hotline.Share.Enums.FlowEngine;
 using Hotline.Share.Enums.Settings;
 using Hotline.Share.Tools;
 using Hotline.Users;
+using MediatR;
 using Microsoft.AspNetCore.Http;
 using Microsoft.AspNetCore.Mvc;
 using Microsoft.AspNetCore.Mvc.Testing;
@@ -31,8 +38,11 @@ public class OrderControllerTest : TestBase
     private readonly OrderController _orderController;
     private readonly IRepository<Hotspot> _hotspotRepository;
     private readonly IOrderRepository _orderRepository;
+    private readonly OrderServiceMock _orderServiceMock;
+    private readonly IRepository<OrderPublish> _orderPublishRepository;
+    private readonly INotificationHandler<EndWorkflowNotify> _orderPublishEndWorkflowHandler;
 
-    public OrderControllerTest(IAccountRepository accountRepository, IRepository<Role> roleRepository, UserController userController, IRepository<Hotspot> hotspotRepository, OrderController orderController, IOrderRepository orderRepository, IServiceScopeFactory scopeFactory, IRepository<User> userRepository) : base(accountRepository, roleRepository, userController, scopeFactory, userRepository)
+    public OrderControllerTest(IAccountRepository accountRepository, IRepository<Role> roleRepository, UserController userController, IRepository<Hotspot> hotspotRepository, OrderController orderController, IOrderRepository orderRepository, IServiceScopeFactory scopeFactory, IRepository<User> userRepository, OrderServiceMock orderServiceMock, IRepository<OrderPublish> orderPublishRepository, INotificationHandler<EndWorkflowNotify> orderPublishEndWorkflowHandler) : base(accountRepository, roleRepository, userController, scopeFactory, userRepository)
     {
         _hotspotRepository = hotspotRepository;
         _orderController = orderController;
@@ -41,34 +51,112 @@ public class OrderControllerTest : TestBase
             HttpContext = new DefaultHttpContext()
         };
         _orderRepository = orderRepository;
+        _orderServiceMock = orderServiceMock;
+        _orderPublishRepository = orderPublishRepository;
+        _orderPublishEndWorkflowHandler = orderPublishEndWorkflowHandler;
     }
 
+
+    /// <summary>
+    /// 创建工单并派送给派单员
+    /// </summary>
+    /// <returns></returns>
     [Fact]
-    public async Task GetOrderHandleTimeConfigByAcceptType_Test()
+    public async Task ToPaiDanYuan_Test()
     {
-        var result = await _orderController.GetOrderHandleTimeConfigByAcceptType(Share.Enums.FlowEngine.EFlowDirection.OrgToCenter, "10");
-        result.Count.ShouldBe(1);
-        result.TimeText.ShouldBe("1个工作日");
-        result.TimeType.ShouldBe(ETimeType.WorkDay);
+        SetZuoXi();
+        var order = _orderServiceMock.CreateOrder()
+            .办理到派单员()
+            .GetCreateResult();
+        order.ShouldNotBeNull();
     }
 
+    /// <summary>
+    /// 验证中心直办工单归档后自动发布
+    /// 是否推诿, 是否不积极
+    /// </summary>
+    /// <returns></returns>
     [Fact]
-    public async Task CreateOrder_Test()
+    public async Task AutoPublish_Test()
     {
-        await SetPaiDanYuan();
-        var orderDto = _fixture.Create<AddOrderDto>();
-        var hotspot = await _hotspotRepository.Queryable()
-            .OrderByDescending(m => m.CreationTime)
-            .FirstAsync();
-        orderDto.HotspotId = hotspot.Id;
-        orderDto.HotspotName = hotspot.HotSpotName;
-        orderDto.HotspotSpliceName = hotspot.HotSpotFullName;
-        orderDto.Files = new List<FileDto>();
+        SetZuoXi();
+        var order = _orderServiceMock.CreateOrder()
+            .办理到派单员()
+            .办理到归档(SetPaiDanYuan, data => 
+            {
+                data.IsEvasive = true;
+                data.IsInactively = true;
+            })
+            .GetCreateResult();
+        var publish = await _orderPublishRepository.GetAsync(m => m.No == order.No);
+        publish.ShouldNotBeNull(order.No);
+        var orderEntity = await _orderRepository.GetAsync(order.Id);
+        orderEntity.IsEvasive.ShouldBeTrue();
+        orderEntity.IsInactively.ShouldBeTrue();
+    }
 
-        var orderResult = await _orderController.Add(orderDto);
-        var orderId = orderResult.ToJson().FromJson<OrderDto>().Id;
-        var order = await _orderRepository.GetAsync(orderId);
+    [Fact]
+    public void Set一级部门_Test()
+    {
+        Set一级部门();
+    }
+
+    [Fact]
+    public async Task IsStepUrgent_True_Test()
+    {
+        SetZuoXi();
+        var order = _orderServiceMock.CreateOrder()
+            .办理到派单员()
+            .办理到归档(SetPaiDanYuan, data => data.IsStepUrgent = true, workflow =>
+            {
+                workflow.BusinessType = EBusinessType.DepartmentLeader;
+                workflow.Opinion = "测试保存勾选紧急";
+            })
+            .GetCreateResult();
         order.ShouldNotBeNull();
-        order.CreatorId.ShouldBe(TestSessionConstants.UserId);
+        order.Id.ShouldNotBeNull();
+        var orderDto = await _orderController.Get(order.Id);
+        orderDto.IsStepUrgent.ShouldBeTrue();
+        orderDto.IsStepUrgenText.ShouldBe("紧急");
+    }
+
+    /// <summary>
+    /// 测试在办理的过程中能正确保存 
+    /// 1.与市民电话联系
+    /// 2.已赴现场处置
+    /// 3.其他
+    /// 4.其他原因
+    /// 5.经办人
+    /// 6.经办人电话
+    /// 7.沟通时间
+    /// 8.沟通地点
+    /// </summary>
+    /// <returns></returns>
+    [Fact]
+    public async Task CreateOrder_Test()
+    {
+        SetZuoXi();
+        var order = _orderServiceMock.CreateOrder()
+            .办理到一级部门()
+            .办理到二级部门(Set一级部门)
+            .办理一级部门汇总(Set二级部门)
+            .GetCreateResult();
+        order.Id.ShouldNotBeNull();
+        var orderEntity = await _orderRepository.GetAsync(order.Id);
+        orderEntity.RealCommunicationAddress.ShouldNotBeNull();
+        orderEntity.RealCommunicationTime.ShouldNotBeNull();
+        orderEntity.RealContactLocale.ShouldNotBeNull();
+        orderEntity.RealContactLocale.ShouldBe(true);
+        orderEntity.RealHandlerName.ShouldNotBeNull();
+        orderEntity.RealHandlerPhone.ShouldNotBeNull();
+    }
+
+    [Fact]
+    public async Task GetOrderHandleTimeConfigByAcceptType_Test()
+    {
+        var result = await _orderController.GetOrderHandleTimeConfigByAcceptType(Share.Enums.FlowEngine.EFlowDirection.OrgToCenter, "10");
+        result.Count.ShouldBe(1);
+        result.TimeText.ShouldBe("1个工作日");
+        result.TimeType.ShouldBe(ETimeType.WorkDay);
     }
 }

+ 1 - 1
src/Hotline.Application.Tests/Controller/PushMessageControllerTest.cs

@@ -29,7 +29,7 @@ public class PushMessageControllerTest
         _userRepository = userRepository;
     }
 
-    [Fact]
+    //[Fact]
     public async Task SendMessage_Test()
     {
         try

+ 3 - 2
src/Hotline.Application.Tests/Domain/OrderVisitDomainServiceTest.cs

@@ -30,7 +30,7 @@ public class OrderVisitDomainServiceTest
         _orderRepository = orderRepository;
     }
 
-    [Fact]
+    //[Fact]
     public async Task UpdateSmsReplyDefault_Test()
     {
         var visit = await _orderVisitRepository
@@ -78,7 +78,7 @@ public class OrderVisitDomainServiceTest
         }
     }
 
-    [Fact]
+    //[Fact]
     public async Task OnSmsUpdate_Test()
     {
         var data = new Message() { IsSmsReply = true, SmsReplyContent = "1", PushBusiness = EPushBusiness.VisitSms };
@@ -99,6 +99,7 @@ public class OrderVisitDomainServiceTest
             .Where(m => m.VisitState == EVisitState.SMSVisiting)
             .OrderByDescending(m => m.CreationTime)
             .FirstAsync();
+        if (visit == null) return;
         visit.ShouldNotBeNull("缺少测试数据");
 
         var message = new MessageDto { ExternalId = visit.Id, IsSmsReply = true, SmsReplyContent = content };

+ 12 - 28
src/Hotline.Application.Tests/Domain/YiBinExpireTimeTest.cs

@@ -40,7 +40,7 @@ public class YiBinExpireTimeTest
     }
 
     [Theory]
-    [InlineData("2024-09-25 13:16:33", 5, "15", "2024-09-29 15:07:40", "5个工作日")]
+    [InlineData("2024-09-25 13:16:33", 5, "15", "2024-10-08 13:16:33", "5个工作日")]
     public async Task CalcEndTimeWorkDayDelay_Test(string begin, int count, string busCode, string expiredTime, string timeText)
     {
         var beginTime = DateTime.Parse(begin);
@@ -66,36 +66,20 @@ public class YiBinExpireTimeTest
     }
 
     [Theory]
-    [InlineData("2024-09-04 14:00:00", "CenterToOrg", "08dccc8f-37b0-40d8-8112-1afb2230c5a3", "2024-09-05 14:00:00")]
-    [InlineData("2024-10-03 09:23:46", "CenterToCenter", "08dcce43-1b95-4a12-813a-b48d7f4ec3bd", "2024-10-14 9:00:00")]
-    public async Task CalcExpiredTime_Test(string beginTxt, string flowTxt, string orderId, string expected)
+    [InlineData("2024-09-04 14:00:00", "CenterToOrg", true, "2024-09-05 14:00:00")]
+    [InlineData("2024-10-03 09:23:46", "CenterToCenter", false, "2024-10-14 9:00:00")]
+    public async Task CalcExpiredTime_Test(string beginTxt, string flowTxt, bool is24Hour , string expected)
     {
         var beginTime = DateTime.Parse(beginTxt);
-        if (orderId.Equals("08dccc8f-37b0-40d8-8112-1afb2230c5a3"))
-            await InitOrderData(orderId);
-        var order = await _orderRepository.Queryable().Where(m => m.Id == orderId).FirstAsync();
         Enum.TryParse(flowTxt, out EFlowDirection flow);
-        var time = await _calcExpireTime.CalcExpiredTime(beginTime, flow, order.Adapt<OrderTimeClacInfo>());
+        var inDto = new OrderTimeClacInfo
+        {
+            Is24HoursComplete = is24Hour,
+            FlowDirection = flow,
+            AcceptTypeCode = "20"
+        };
+        var time = await _calcExpireTime.CalcExpiredTime(beginTime, flow,inDto);
         time.ShouldNotBeNull();
         time.ExpiredTime.ShouldBe(DateTime.Parse(expected));
     }
-
-    public async Task InitOrderData(string orderId)
-    {
-        var order = await _orderRepository.Queryable().Where(m => m.Id == orderId).FirstAsync();
-        if (order.Is24HoursComplete) return;
-        order.Is24HoursComplete = true;
-        await _orderRepository.UpdateAsync(order);
-    }
-
-    [Fact]
-    public async Task InitExpireTime_Test()
-    {
-        var entity = new TimeLimitSetting() { BusCode = "YQ", BusName = "疫情", TimeType = ETimeType.WorkDay, TimeValue = 2, Percentage = 80, PercentageOne = 50 };
-        if (await _timeLimitSettingRepository.Queryable().Where(m => m.BusCode == entity.BusCode).FirstAsync() == null)
-            await _timeLimitSettingRepository.AddAsync(entity);
-        entity = new TimeLimitSetting() { BusCode = "24", BusName = "24小时", TimeType = ETimeType.Hour, TimeValue = 24, Percentage = 80, PercentageOne = 50 };
-        if (await _timeLimitSettingRepository.Queryable().Where(m => m.BusCode == entity.BusCode).FirstAsync() == null)
-            await _timeLimitSettingRepository.AddAsync(entity);
-    }
-}
+  }

+ 26 - 55
src/Hotline.Application.Tests/Domain/ZiGongExpireTimeTest.cs

@@ -1,10 +1,13 @@
-using Hotline.Orders;
+using AngleSharp.Text;
+using Hotline.Orders;
 using Hotline.Settings;
 using Hotline.Settings.TimeLimitDomain;
 using Hotline.Settings.TimeLimitDomain.Repository;
 using Hotline.Share.Dtos.Settings;
 using Hotline.Share.Enums.FlowEngine;
+using Hotline.Share.Enums.Order;
 using Hotline.Share.Enums.Settings;
+using Hotline.Share.Tools;
 using Mapster;
 using Shouldly;
 using XF.Domain.Repository;
@@ -25,65 +28,31 @@ public class ZiGongExpireTimeTest
         _timeLimitSettingInventoryRepository = timeLimitSettingInventoryRepository;
     }
 
-    //[Theory]
-    //// 24小时件
-    //[InlineData("24小时件", "2024-09-04 14:00:00", "CenterToOrg", "08dccc8f-37b0-40d8-8112-1afb2230c5a3", "2024-09-05 14:00:00")]
-    //// 疫情件
-    //[InlineData("疫情件", "2024-09-05 14:01:01", "CenterToOrg", "08dccd5c-9bda-4e7d-8d63-82039dcfbde7", "2024-09-09 14:01:01")]
-    //public async Task CalcExpiredTime_Test(string tip, string beginTxt, string flowTxt, string orderId, string expected)
-    //{
-    //    var beginTime = DateTime.Parse(beginTxt);
-    //    if (orderId.Equals("08dccc8f-37b0-40d8-8112-1afb2230c5a3"))
-    //        await InitOrderData(orderId);
-    //    var order = await _orderRepository.Queryable().Where(m => m.Id == orderId).FirstAsync();
-    //    Enum.TryParse(flowTxt, out EFlowDirection flow);
-    //    var time = await _ziGongExpireTimeLimit.CalcExpiredTime(beginTime, flow, order.Adapt<OrderTimeClacInfo>());
-    //    time.ShouldNotBeNull();
-    //    time.ExpiredTime.ShouldBe(DateTime.Parse(expected), $"{tip} 期满时间错误");
-    //}
-
     [Theory]
-    [InlineData("企业咨询件单元测试", "2024-09-04 14:00:00", "CenterToOrg", "2024-09-05 14:00:00")]
-    [InlineData("企业建议件单元测试", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-09 14:00:00")]
-    [InlineData("企业求助件单元测试", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-09 14:00:00")]
-    [InlineData("企业表扬件单元测试", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-09 14:00:00")]
-    [InlineData("企业举报件单元测试", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-09 14:00:00")]
-    [InlineData("企业投诉件单元测试", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-09 14:00:00")]
-    [InlineData("四川省12345咨询件单测试", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-06 14:00:00")]
-    [InlineData("四川省12345建议件单测试", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-10 14:00:00")]
-    [InlineData("中心到中心24小时", "2024-09-12 14:00:00", "CenterToCenter", "2024-09-13 14:00:00")]
-    public async Task CalcExpiredTime_Test(string title, string beginTxt, string flowTxt, string expected)
+    [InlineData("企业咨询件单元测试",     false, "Enterprise","10","互联网", "2024-09-04 14:00:00", "CenterToOrg", "2024-09-05 14:00:00")]
+    [InlineData("企业建议件单元测试",     false, "Enterprise","15","互联网", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-09 14:00:00")]
+    [InlineData("企业求助件单元测试",     false, "Enterprise","20","互联网", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-09 14:00:00")]
+    [InlineData("企业表扬件单元测试",     false, "Enterprise","25","互联网", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-09 14:00:00")]
+    [InlineData("企业举报件单元测试",     false, "Enterprise","30","互联网", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-09 14:00:00")]
+    [InlineData("企业投诉件单元测试",     false, "Enterprise","35","互联网", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-09 14:00:00")]
+    [InlineData("四川省12345咨询件单测试",false, "Citizen", "10","四川省12345", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-06 14:00:00")]
+    [InlineData("四川省12345建议件单测试",false, "Citizen", "15","四川省12345", "2024-09-05 14:00:00", "CenterToOrg", "2024-09-10 14:00:00")]
+    [InlineData("中心到中心24小时",       true, "Citizen", "10","四川省12345", "2024-09-12 14:00:00", "CenterToCenter", "2024-09-13 14:00:00")]
+    public async Task CalcExpiredTime_Test(string title, bool is24,  string identityType,string acceptTypeCode, string sourceChannel, string beginTxt, string flowTxt, string expected)
     {
         var beginTime = DateTime.Parse(beginTxt);
-        var order = await _orderRepository.Queryable().Where(m => m.Title == title).FirstAsync();
-        order.ShouldNotBeNull($"{title} 测试数据不存在");
         Enum.TryParse(flowTxt, out EFlowDirection flow);
-        var time = await _ziGongExpireTimeLimit.CalcExpiredTime(beginTime, flow, order.Adapt<OrderTimeClacInfo>());
+        var inDto = new OrderTimeClacInfo 
+        {
+            IdentityType = identityType.ToEnum<EIdentityType>(),
+            AcceptTypeCode = acceptTypeCode,
+            SourceChannel  = sourceChannel,
+            FlowDirection = flow,
+            Is24HoursComplete = is24
+        };
+        var time = await _ziGongExpireTimeLimit.CalcExpiredTime(beginTime, flow, inDto);
         time.ShouldNotBeNull();
-        time.ExpiredTime.ShouldBe(DateTime.Parse(expected), $"{title} 期满时间错误 AcceptTypeCode:{order.AcceptTypeCode}");
-        time.TimeText.ShouldBe(order.Content, $"{title} 内容结果比对失败 AcceptTypeCode:{order.AcceptTypeCode}");
-    }
-
-    [Theory]
-    [InlineData("求助三个工作日", "2024-09-12 22:01:28", "CenterToOrg", "2024-09-20 08:30:00")]
-    [InlineData("string", "2024-09-12 22:01:28", "CenterToOrg", "2024-09-20 08:30:00")]
-    public async Task CalcExpiredTime_Release_Test(string title, string beginTxt, string flowTxt, string expected)
-    {
-        var beginTime = DateTime.Parse(beginTxt);
-        var order = await _orderRepository.Queryable().Where(m => m.Title == title).FirstAsync();
-        order.ShouldNotBeNull($"{title} 测试数据不存在");
-        Enum.TryParse(flowTxt, out EFlowDirection flow);
-        var time = await _ziGongExpireTimeLimit.CalcExpiredTime(beginTime, flow, order.Adapt<OrderTimeClacInfo>());
-        time.ShouldNotBeNull();
-        time.ExpiredTime.ShouldBe(DateTime.Parse(expected), $"{title} 期满时间错误 AcceptTypeCode:{order.AcceptTypeCode}");
-    }
-
-    public async Task InitOrderData(string orderId)
-    {
-        var order = await _orderRepository.Queryable().Where(m => m.Id == orderId).FirstAsync();
-        if (order.Is24HoursComplete) return;
-        order.Is24HoursComplete = true;
-        await _orderRepository.UpdateAsync(order);
+        time.ExpiredTime.ShouldBe(DateTime.Parse(expected), $"{title} 期满时间错误 AcceptTypeCode:{acceptTypeCode}");
     }
 
     [Theory]
@@ -94,6 +63,7 @@ public class ZiGongExpireTimeTest
     [InlineData("20", "IdentityType", "Enterprise", false)]
     public async Task InitTimeLimitData_Test(string busCode, string name, string value, bool isCommon)
     {
+        return;
         var attributeEntity = new TimeLimitSettingAttribute { BusCode = busCode, Name = name, Value = value, IsCommon = isCommon };
         var dataEntity = await _timeLimitSettingAttributeRepository.GetAsync(attributeEntity.BusCode, attributeEntity.Name, attributeEntity.Value);
         if (dataEntity is null) await _timeLimitSettingAttributeRepository.AddAsync(attributeEntity);
@@ -121,6 +91,7 @@ public class ZiGongExpireTimeTest
     [InlineData("", "SourceChannel", "中国政府网", "WorkDay", 3, "中国政府网 '非咨询' 件3个工作日")]
     public async Task InitTimeLimitInventory_Test(string busCode, string name, string value, string timeType, int timeValue, string remark)
     {
+        return;
         var attributeEntity = new TimeLimitSettingAttribute { BusCode = busCode, Name = name, Value = value, IsCommon = false };
         var dataEntity = await _timeLimitSettingAttributeRepository.GetAsync(attributeEntity.BusCode, attributeEntity.Name, attributeEntity.Value);
         if (dataEntity is null) await _timeLimitSettingAttributeRepository.AddAsync(attributeEntity);

+ 14 - 0
src/Hotline.Application.Tests/Dto/CreateOrderOutDto.cs

@@ -0,0 +1,14 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+
+namespace Hotline.Application.Tests.Dto;
+public class CreateOrderOutDto
+{
+    public string Id { get; set; }
+    public string No { get; set; }
+    public string Password { get; set; }
+    public string CallId { get; set; }
+}

+ 0 - 4
src/Hotline.Application.Tests/Hotline.Application.Tests.csproj

@@ -47,8 +47,4 @@
     <Using Include="Xunit" />
   </ItemGroup>
 
-  <ItemGroup>
-    <Folder Include="Authentications\" />
-  </ItemGroup>
-
 </Project>

+ 4 - 0
src/Hotline.Application.Tests/Infrastructure/TestSettingConstants.cs

@@ -8,6 +8,9 @@ namespace Hotline.Application.Tests.Infrastructure;
 public static class TestSettingConstants
 {
     public const string PaiDanYuanAccountName = "UnitTestPDY";
+    public const string FirstOrgAccountName = "cs";
+    public const string SecondOrgAccountName = "cs21";
+    public const string BanZhangAccountName = "UnitTestBZ";
 }
 
 public static class TestSessionConstants
@@ -21,4 +24,5 @@ public static class TestSessionConstants
     public static string UserName = "";
 
     public static string OrgId = "";
+
 }

+ 53 - 0
src/Hotline.Application.Tests/Mock/KnowledgeServiceMock.cs

@@ -0,0 +1,53 @@
+using Hotline.Api.Controllers;
+using Hotline.Application.FlowEngine;
+using Hotline.Application.Tests.Infrastructure;
+using Hotline.FlowEngine.WorkflowModules;
+using Hotline.Identity.Accounts;
+using Hotline.Share.Dtos.FlowEngine;
+using Hotline.Share.Dtos.Knowledge;
+using Hotline.Share.Enums.FlowEngine;
+using Hotline.Share.Tools;
+using Hotline.Users;
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+using XF.Domain.Repository;
+
+namespace Hotline.Application.Tests.Mock;
+public class KnowledgeServiceMock
+{
+    private readonly KnowledgeController _knowledgeController;
+    private string KnowledgeId;
+    private readonly IWorkflowApplication _workflowApplication;
+    private readonly IRepository<Account> _accountRepository;
+
+    public KnowledgeServiceMock(KnowledgeController knowledgeController, IWorkflowApplication workflowApplication, IRepository<Account> accountRepository)
+    {
+        _knowledgeController = knowledgeController;
+        _workflowApplication = workflowApplication;
+        _accountRepository = accountRepository;
+    }
+
+    public KnowledgeServiceMock 创建并审核知识()
+    {
+        var json = "{\"data\":{\"attribution\":\"中心知识库\",\"isPublic\":true,\"keywords\":[\"08dcfe29-01a5-4b6e-8b47-640d75aa8b23\",\"08dcfe29-01a7-4488-8412-65907c33576d\",\"08dcfe29-01a8-4217-84ae-45eda2481c9b\"],\"keywordsName\":\"张继科扥,金卡,希望\",\"knowledgeTypeId\":[\"08dc903c-aad8-4515-89f3-1ae2d6961b90\"],\"hotspotId\":\"\",\"knowledges\":[],\"files\":[],\"template\":\"\",\"content\":\"<p style=\\\"line-height: 2;\\\"><span style=\\\"font-size: 20px; font-family: 仿宋;\\\">你张继科扥金卡希望智啊</span></p>\",\"sourceOrganizeId\":\"\",\"title\":\"你张继科扥金卡希望智啊\",\"knowledgeType\":[{\"KnowledgeTypeName\":\"类型11\",\"KnowledgeTypeId\":\"08dc903c-aad8-4515-89f3-1ae2d6961b90\",\"KnowledgeTypeSpliceName\":\"类型11\"}]},\"workflow\":{\"isPass\":true,\"opinion\":\"123\",\"nextStepCode\":\"da354c75-d8f7-4c8c-af6f-1e69fb09c343\",\"nextStepName\":\"流程节点\",\"backToCountersignEnd\":false,\"nextHandlers\":[{\"userId\":\"08dc3c21-8e9b-4387-8306-6cc32d127397\",\"username\":\"班长2\",\"orgId\":\"001\",\"orgName\":\"市民热线服务中心\",\"roleId\":\"zhongxinlingdao\",\"roleName\":\"中心领导\",\"key\":\"08dc3c21-8e9b-4387-8306-6cc32d127397\",\"value\":\"班长2\"}],\"nextMainHandler\":\"08dc3c21-8e9b-4387-8306-6cc32d127397\",\"isSms\":false,\"isStartCountersign\":false,\"stepId\":null,\"nextHandler\":{},\"workflowId\":\"\",\"handlerType\":0,\"businessType\":0,\"flowDirection\":2,\"external\":{},\"files\":[]}}";
+        var inDto = json.FromJson<AddStartFlowDto>();
+        var nextStep = _workflowApplication.GetStartStepsAsync(WorkflowModuleConsts.KnowledgeAdd, CancellationToken.None).GetAwaiter().GetResult();
+        inDto.Data.Title = "单元测试" + DateTime.Now.ToString("yyyyMMddhhmmss");
+        inDto.Workflow.FlowDirection = EFlowDirection.CenterToCenter;
+        inDto.Workflow.NextStepCode = nextStep.Steps.First().Key;
+        inDto.Workflow.NextStepName = nextStep.Steps.First().Value;
+        var account = _accountRepository.GetAsync(m => m.UserName == TestSettingConstants.BanZhangAccountName).GetAwaiter().GetResult();
+        var handler = nextStep.Steps.First().Items.Where(m => m.Username == account.Name).First();
+        inDto.Workflow.NextHandlers.Add(handler);
+        KnowledgeId = _knowledgeController.AddKnowledge(inDto).GetAwaiter().GetResult();
+        return this;
+    }
+
+    public string GetKnowledgeId()
+    {
+        return KnowledgeId;
+    }
+}

+ 239 - 0
src/Hotline.Application.Tests/Mock/OrderServiceMock.cs

@@ -0,0 +1,239 @@
+using Hotline.Api.Controllers;
+using Hotline.Application.Tests.Dto;
+using Hotline.Orders;
+using Hotline.Repository.SqlSugar.Orders;
+using Hotline.Share.Dtos.FlowEngine;
+using Hotline.Share.Dtos.Order;
+using Hotline.Share.Enums.FlowEngine;
+using Hotline.Share.Tools;
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+using XF.Domain.Repository;
+
+namespace Hotline.Application.Tests.Mock;
+public class OrderServiceMock
+{
+    private readonly OrderController _orderController;
+    private CreateOrderOutDto CreateOrderOutDto;
+    private AddOrderDto AddOrderDto;
+    private readonly IRepository<Order> _orderRepository;
+
+    public OrderServiceMock(OrderController orderController, IRepository<Order> orderRepository)
+    {
+        _orderController = orderController;
+        _orderRepository = orderRepository;
+    }
+
+    public OrderServiceMock CreateOrder()
+    {
+        var json = "{\"sourceChannel\":\"因特网\",\"sourceChannelCode\":\"YTW\",\"transferPhone\":null,\"fromPhone\":null,\"acceptorName\":\"单元测试\",\"acceptorStaffNo\":\"\",\"fromName\":\"1233333333\",\"fromGender\":1,\"identityType\":1,\"licenceType\":null,\"licenceTypeCode\":null,\"licenceNo\":null,\"ageRange\":null,\"ageRangeCode\":null,\"contact\":\"12333333333\",\"isSecret\":false,\"acceptSms\":false,\"no\":null,\"title\":\"\",\"hotspotId\":\"1912\",\"eventCategoryId\":null,\"incidentTime\":null,\"incidentPurpose\":null,\"areaCode\":\"519800\",\"city\":\"省内\",\"street\":null,\"isRepeat\":\"false\",\"pushType\":null,\"pushTypeCode\":null,\"content\":\"单元测试内容\",\"duplicateIds\":[],\"duplicateTitle\":null,\"callAddress\":null,\"repeatableEventDetails\":[],\"orderExtension\":null,\"transpond\":false,\"isEnforcementOrder\":false,\"focusOnEventsArr\":[],\"focusOnEvents\":null,\"isFormalistWorkOrder\":false,\"isSensitiveWorkOrders\":false,\"isUrgent\":false,\"isThreePartyConference\":false,\"is24HoursComplete\":false,\"company\":null,\"orderPushTypes\":[],\"acceptType\":\"咨询\",\"acceptTypeCode\":\"10\",\"files\":[],\"hotspotSpliceName\":\"互联互通-转接乐山市12345\",\"hotspotName\":\"转接乐山市12345\",\"hotspotCode\":\"1912\",\"hotspotExternal\":\"19\",\"county\":\"\",\"town\":\"\"}";
+        AddOrderDto = json.FromJson<AddOrderDto>();
+        AddOrderDto.Title = "单元测试" + DateTime.Now.ToString("yyyy-MM-dd hh:mm:ss");
+        CreateOrderOutDto = _orderController.Add(AddOrderDto).GetAwaiter().GetResult().ToJson().FromJson<CreateOrderOutDto>();
+        return this;
+    }
+
+    public CreateOrderOutDto GetCreateResult()
+    {
+        return CreateOrderOutDto;
+    }
+
+    public OrderServiceMock 办理到派单员()
+    { 
+        var stepNextInfo = _orderController.GetFlowStartOptions(CreateOrderOutDto.Id).GetAwaiter().GetResult().ToJson().FromJson<NextStepsDto<NextStepOption>>();
+        var stepInfo = stepNextInfo.Steps.FirstOrDefault(m => m.Value == "派单组");
+        var stepOrg = stepInfo.Items.FirstOrDefault(m => m.Username == "单元测试派单员");
+
+        var handleDto = new StartWorkflowDto<OrderHandleFlowDto>
+        {
+            Data = new OrderHandleFlowDto
+            {
+                OrderId = CreateOrderOutDto.Id,
+            },
+            Workflow = new BasicWorkflowDto
+            {
+                NextHandlers = [stepOrg],
+                NextStepCode = stepInfo.Key,
+                NextStepName = stepInfo.Value,
+                Opinion = "办理到派单组意见",
+                BackToCountersignEnd = false,
+                IsSms = false,
+                IsForwarded = false,
+                HandlerType = EHandlerType.OrgLevel,
+                BusinessType = EBusinessType.Send,
+                FlowDirection = EFlowDirection.CenterToCenter,
+            }
+        };
+        _orderController.StartFlow(handleDto).GetAwaiter().GetResult();
+        return this;
+
+    }
+
+    public OrderServiceMock 办理到一级部门()
+    {
+        var stepNextInfo = _orderController.GetFlowStartOptions(CreateOrderOutDto.Id).GetAwaiter().GetResult().ToJson().FromJson<NextStepsDto<NextStepOption>>();
+        var stepInfo = stepNextInfo.Steps.FirstOrDefault(m => m.Value == "一级部门");
+        var stepOrg = stepInfo.Items.FirstOrDefault(m => m.OrgName == "测试部门");
+
+        var handleDto = new StartWorkflowDto<OrderHandleFlowDto>
+        {
+            Data = new OrderHandleFlowDto
+            {
+                OrderId = CreateOrderOutDto.Id,
+            },
+            Workflow = new BasicWorkflowDto
+            {
+                NextHandlers = [stepOrg],
+                NextStepCode = stepInfo.Key,
+                NextStepName = stepInfo.Value,
+                Opinion = "办理意见",
+                BackToCountersignEnd = false,
+                IsSms = false,
+                IsForwarded = false,
+                HandlerType = EHandlerType.OrgLevel,
+                BusinessType = EBusinessType.Department,
+            }
+        };
+        _orderController.StartFlow(handleDto).GetAwaiter().GetResult();
+        return this;
+    }
+
+    public OrderServiceMock 办理到归档(Action action = null, Action<OrderHandleFlowDto> dataAction = null, Action<NextWorkflowDto> workflowAction = null)
+    {
+        action?.Invoke();
+        var stepNextInfo = _orderController.GetNextStepsWithRecommend(CreateOrderOutDto.Id).GetAwaiter().GetResult().ToJson().FromJson<NextStepsDto<NextStepOption>>();
+        var stepInfo = stepNextInfo.Steps.FirstOrDefault(m => m.Value == "归档");
+        var stepOrg = stepInfo.Items.FirstOrDefault();
+
+        var order = _orderRepository.Get(CreateOrderOutDto.Id);
+
+        var handleDto = new NextWorkflowDto<OrderHandleFlowDto>
+        {
+            Data = new OrderHandleFlowDto
+            {
+                OrderId = CreateOrderOutDto.Id,
+                RealHandlerName = "经办人",
+                RealHandlerPhone = "13666666666",
+                RealContactLocale = true,
+                RealIsContacted = true,
+                IsOther = true,
+                OtherRemark = "其它原因",
+                RealCommunicationAddress = "地点地点地点",
+                RealCommunicationTime = DateTime.Now,
+            },
+            Workflow = new NextWorkflowDto
+            {
+                NextStepCode = "end",
+                NextStepName = "归档",
+                Opinion = "办理到归档",
+                BackToCountersignEnd = false,
+                IsSms = false,
+                IsForwarded = false,
+                HandlerType = EHandlerType.OrgLevel,
+                BusinessType = EBusinessType.Department,
+                FlowDirection = EFlowDirection.OrgToOrg,
+                WorkflowId = order.WorkflowId,
+                StepId = stepNextInfo.StepId,
+            }
+        };
+        dataAction?.Invoke(handleDto.Data);
+        workflowAction?.Invoke(handleDto.Workflow);
+        try
+        {
+            _orderController.Handle(handleDto).GetAwaiter().GetResult();
+        }
+        catch (Exception e)
+        {
+            var msg = e.Message;
+            if (msg.Contains("RealtimeService") == false)
+            {
+                throw;
+            }
+            // ignore
+        }
+        return this;
+    }
+
+    public OrderServiceMock 办理到二级部门(Action action = null)
+    {
+        action?.Invoke();
+        var stepNextInfo = _orderController.GetNextStepsWithRecommend(CreateOrderOutDto.Id).GetAwaiter().GetResult().ToJson().FromJson<NextStepsDto<NextStepOption>>();
+        var stepInfo = stepNextInfo.Steps.FirstOrDefault(m => m.Value == "二级部门");
+        var stepOrg = stepInfo.Items.FirstOrDefault(m => m.OrgName == "测试二级部门");
+
+        var order = _orderRepository.Get(CreateOrderOutDto.Id);
+
+        var handleDto = new NextWorkflowDto<OrderHandleFlowDto>
+        {
+            Data = new OrderHandleFlowDto
+            {
+                OrderId = CreateOrderOutDto.Id,
+            },
+            Workflow = new NextWorkflowDto
+            {
+                NextHandlers = [stepOrg],
+                NextStepCode = stepInfo.Key,
+                NextStepName = stepInfo.Value,
+                Opinion = "办理到二级部门",
+                BackToCountersignEnd = false,
+                IsSms = false,
+                IsForwarded = false,
+                HandlerType = EHandlerType.OrgLevel,
+                BusinessType = EBusinessType.Department,
+                FlowDirection = EFlowDirection.OrgToOrg,
+                WorkflowId = order.WorkflowId,
+                StepId = stepNextInfo.StepId,
+            }
+        };
+        _orderController.Handle(handleDto).GetAwaiter().GetResult();
+        return this;
+    }
+
+    public OrderServiceMock 办理一级部门汇总(Action aciton = null)
+    {
+        aciton?.Invoke();
+        var stepNextInfo = _orderController.GetNextStepsWithRecommend(CreateOrderOutDto.Id).GetAwaiter().GetResult().ToJson().FromJson<NextStepsDto<NextStepOption>>();
+        var stepInfo = stepNextInfo.Steps.FirstOrDefault(m => m.Value == "一级部门汇总");
+        var stepOrg = stepInfo.Items.FirstOrDefault(m => m.OrgName == "测试部门");
+
+        var order = _orderRepository.Get(CreateOrderOutDto.Id);
+
+        var handleDto = new NextWorkflowDto<OrderHandleFlowDto>
+        {
+            Data = new OrderHandleFlowDto
+            {
+                OrderId = CreateOrderOutDto.Id,
+                RealHandlerName = "经办人",
+                RealHandlerPhone = "13666666666",
+                RealContactLocale = true,
+                RealIsContacted = true,
+                IsOther = true,
+                IsEvasive = true,
+                OtherRemark = "其它原因",
+                RealCommunicationAddress = "地点地点地点",
+                RealCommunicationTime = DateTime.Now
+            },
+            Workflow = new NextWorkflowDto
+            {
+                NextHandlers = [stepOrg],
+                NextStepCode = stepInfo.Key,
+                NextStepName = stepInfo.Value,
+                Opinion = "办理到一级部门汇总",
+                BackToCountersignEnd = false,
+                IsSms = false,
+                IsForwarded = false,
+                HandlerType = EHandlerType.AssignedUser,
+                BusinessType = EBusinessType.Department,
+                FlowDirection = EFlowDirection.OrgToOrg,
+                StepType = EStepType.Summary,
+                WorkflowId = order.WorkflowId,
+                StepId = stepNextInfo.StepId,
+            }
+        };
+        _orderController.Handle(handleDto).GetAwaiter().GetResult();
+        return this;
+    }
+}

+ 25 - 0
src/Hotline.Application.Tests/Startup.cs

@@ -38,6 +38,8 @@ using Hotline.Identity.Accounts;
 using Hotline.Application.Tests.Controller;
 using Hotline.Application.Tests.Infrastructure;
 using Hotline.Authentications;
+using Hotline.FlowEngine.Notifications;
+using Hotline.Application.Handlers.FlowEngine;
 
 namespace Hotline.Application.Tests;
 public class Startup
@@ -117,6 +119,7 @@ public class Startup
                         callCenterConfiguration.TianRun.Username,
                         callCenterConfiguration.TianRun.Password);
             services.RegisterMediatR(appConfiguration);
+            services.RegisterSignalR(configuration);
 
             // services.AddSenparcWeixinServices(configuration);
             AppDomain.CurrentDomain.GetAssemblies().ToList().SelectMany(d => d.GetTypes())
@@ -146,10 +149,14 @@ public class Startup
             services.AddScoped<IExportApplication, ExportApplication>();
             services.AddScoped<OrderController>();
             services.AddScoped<UserController>();
+            services.AddScoped<KnowledgeController>();
             services.AddScoped<PushMessageController>();
             services.AddScoped<ISessionContext, DefaultHttpContextAccessor>();
             services.AddScoped<ISessionContextProvider, SessionContextProvider>();
             services.AddScoped<ICallApplication, XingTangCallApplication>();
+            services.AddScoped<XingTangCallApplication>();
+            services.AddScoped<OrderServiceMock>();
+            services.AddScoped<KnowledgeServiceMock>();
             //ServiceLocator.Instance = services.BuildServiceProvider();
         }
 
@@ -180,6 +187,24 @@ public static class StartupHelper
         return services;
     }
 
+    /// <summary>
+    /// SignalR
+    /// </summary>
+    /// <param name="services"></param>
+    /// <param name="configuration"></param>
+    /// <returns></returns>
+    public static IServiceCollection RegisterSignalR(this IServiceCollection services, IConfiguration configuration)
+    {
+        var cacheConfig = configuration.GetRequiredSection("Cache").Get<CacheOptions>();
+        services.AddSignalR().AddStackExchangeRedis($"{cacheConfig.Host}:{cacheConfig.Port}", options =>
+        {
+            options.Configuration.ChannelPrefix = "Hotline:signalr:";
+            options.Configuration.Password = cacheConfig.Password;
+            options.Configuration.DefaultDatabase = cacheConfig.Database;
+        });
+        return services;
+    }
+
     public static IServiceCollection AddTestMq(this IServiceCollection services, IConfiguration configuration)
     {
         MqConfiguration mqConfiguration = configuration.GetSection("MqConfiguration").Get<MqConfiguration>();

+ 28 - 13
src/Hotline.Application.Tests/TestBase.cs

@@ -36,28 +36,43 @@ public class TestBase
         _userRepository = userRepository;
     }
 
-    public async Task SetPaiDanYuan()
+    public void SetPaiDanYuan()
     {
-        await SetOperator("派单员", "市民热线服务中心", "单元测试派单员", "001", "13408389849", EUserType.Normal, TestSettingConstants.PaiDanYuanAccountName);
+        SetOperator("派单员", "市民热线服务中心", "单元测试派单员", "001", "13408389849", EUserType.Normal, TestSettingConstants.PaiDanYuanAccountName);
     }
 
-    public async Task SetZuoXi()
+    public void Set一级部门()
     {
-        await SetOperator("坐席", "市民热线服务中心", "单元测试派单员", "001", "13408389849", EUserType.Seat, TestSettingConstants.PaiDanYuanAccountName);
+        SetOperator("部门经办人", "测试部门", "测试", "001094", "13408389849", EUserType.Seat, TestSettingConstants.FirstOrgAccountName);
     }
 
-    private async Task SetOperator(string displayName, string fullOrgName, string name, string orgId, string phoneNo, EUserType userType, string userName)
+    public void Set二级部门()
     {
-        var account = await _accountRepository.GetExtAsync(
+        SetOperator("部门经办人", "测试部门/测试二级部门", "cs21", "001094001", "13408389849", EUserType.Seat, TestSettingConstants.SecondOrgAccountName);
+    }
+
+    public void Set班长()
+    {
+        SetOperator("中心班长", "市民热线服务中心", "单元测试班长", "001", "13408389849", EUserType.Normal, TestSettingConstants.BanZhangAccountName);
+    }
+
+    public void SetZuoXi()
+    {
+        SetOperator("坐席", "市民热线服务中心", "单元测试派单员", "001", "13408389849", EUserType.Seat, TestSettingConstants.PaiDanYuanAccountName);
+    }
+
+    private void SetOperator(string displayName, string fullOrgName, string name, string orgId, string phoneNo, EUserType userType, string userName)
+    {
+        var account = _accountRepository.GetExtAsync(
             d => d.UserName == userName,
-            d => d.Includes(x => x.Roles));
+            d => d.Includes(x => x.Roles)).GetAwaiter().GetResult();
 
         if (account == null)
         {
-            var roleId = await _roleRepository.Queryable()
+            var roleId = _roleRepository.Queryable()
                 .Where(m => m.DisplayName == displayName)
                 .Select(m => m.Id)
-                .FirstAsync();
+                .First();
             var newUser = new AddUserDto
             {
                 FullOrgName = fullOrgName,
@@ -69,13 +84,13 @@ public class TestBase
                 UserType = userType,
                 UserName = userName
             };
-            var accountId = await _userController.Add(newUser);
+            var accountId = _userController.Add(newUser).GetAwaiter().GetResult();
             TestSessionConstants.UserId = accountId;
-            account = await _accountRepository.GetExtAsync(
+            account = _accountRepository.GetExtAsync(
                 d => d.UserName == userName,
-                d => d.Includes(x => x.Roles));
+                d => d.Includes(x => x.Roles)).GetAwaiter().GetResult();
         }
-        var user = await _userRepository.GetAsync(account.Id);
+        var user = _userRepository.GetAsync(account.Id).GetAwaiter().GetResult();
         TestSessionConstants.UserId = account.Id;
         TestSessionConstants.Roles = account.Roles.Select(m => m.Id).ToArray();
         TestSessionConstants.UserName = account.UserName;

+ 16 - 5
src/Hotline.Application/ExportWord/WordHelper.cs

@@ -1,6 +1,7 @@
 using DocumentFormat.OpenXml.Packaging;
 using DocumentFormat.OpenXml.Wordprocessing;
 using Hotline.Share.Dtos.ExportWord;
+using Hotline.Share.Tools;
 using HtmlToOpenXml;
 using OpenHtmlToPdf;
 
@@ -30,14 +31,24 @@ namespace Hotline.Application.ExportWord
                 mainPart.Document = new Document(new Body());
 
                 HtmlConverter converter = new HtmlConverter(mainPart);
-                var paragraphs = converter.Parse(htmlContent);
+                var validator = new HtmlImageValidator();
+                string validatedHtml = validator.ValidateAndReplaceImagesAsync(htmlContent);
+                try
+                {
+                    var paragraphs = converter.Parse(validatedHtml);
+
+                    foreach (var paragraph in paragraphs)
+                    {
+                        mainPart.Document.Body.Append(paragraph);
+                    }
 
-                foreach (var paragraph in paragraphs)
+                    mainPart.Document.Save();
+                }
+                catch (Exception e)
                 {
-                    mainPart.Document.Body.Append(paragraph);
+                    var msg = e.Message;
+                    // ignore
                 }
-
-                mainPart.Document.Save();
             }
 
             memoryStream.Position = 0;

+ 0 - 6
src/Hotline.Application/FlowEngine/IWorkflowApplication.cs

@@ -63,12 +63,6 @@ namespace Hotline.Application.FlowEngine
         Task HandleToEndAsync(string workflowId, string opinion, List<FileDto> files,
             EReviewResult reviewResult = EReviewResult.Unknown, CancellationToken cancellationToken = default);
 
-        /// <summary>
-        /// 跳转至结束节点(无视流程模板配置以及当前办理对象)
-        /// </summary>
-        Task JumpToEndAsync(ISessionContext current,string workflowId, string opinion, List<FileDto> files, DateTime? expiredTime,
-            EReviewResult reviewResult = EReviewResult.Unknown, CancellationToken cancellationToken = default);
-
         ////////
 
         /// <summary>

+ 17 - 92
src/Hotline.Application/FlowEngine/WorkflowApplication.cs

@@ -442,84 +442,6 @@ public class WorkflowApplication : IWorkflowApplication, IScopeDependency
         await NextAsync(dto, cancellationToken: cancellationToken);
     }
 
-    /// <summary>
-    /// 跳转至结束节点(无视流程模板配置以及当前办理对象,直接跳至结束节点)
-    /// </summary>
-    public async Task JumpToEndAsync(ISessionContext current, string workflowId, string opinion, List<FileDto> files, DateTime? expiredTime,
-        EReviewResult reviewResult = EReviewResult.Unknown, CancellationToken cancellationToken = default)
-    {
-        var workflow = await _workflowDomainService.GetWorkflowAsync(workflowId, withDefine: true, withSteps: true,
-            withTraces: true, cancellationToken: cancellationToken);
-        var endStepDefine = workflow.WorkflowDefinition.FindEndStepDefine();
-        if (endStepDefine is null)
-            throw new UserFriendlyException("未正确配置结束节点");
-
-        //var currentStep = workflow.GetActualStep();
-        //if (currentStep is null)
-        //    throw new UserFriendlyException("未找到实际办理节点");
-
-        var dto = new BasicWorkflowDto
-        {
-            NextStepCode = endStepDefine.Code,
-            NextStepName = endStepDefine.Name,
-            FlowDirection = EFlowDirection.OrgToFile,
-            BusinessType = endStepDefine.BusinessType,
-            ReviewResult = reviewResult,
-            Opinion = opinion,
-            Files = files
-        };
-
-        var unhandleSteps = workflow.Steps
-            .Where(d => d.Status != EWorkflowStepStatus.Handled).ToList();
-        var unhandleTraces = workflow.Traces
-            .Where(d => d.Status != EWorkflowStepStatus.Handled).ToList();
-
-        //get currentStep
-        var currentStep = unhandleSteps.MaxBy(d => d.CreationTime)
-                          ?? workflow.Steps.MaxBy(d => d.CreationTime);
-
-        foreach (var step in unhandleSteps)
-        {
-            await _workflowDomainService.HandleStepAsync(step, workflow, dto, null, null, cancellationToken);
-            if (step.IsStartCountersign)
-                step.CountersignEnd();
-
-            var trace = unhandleTraces.First(d => d.StepId == step.Id);
-            _mapper.Map(dto, trace);
-            _mapper.Map(step, trace);
-        }
-
-        await _workflowStepRepository.UpdateRangeAsync(unhandleSteps, cancellationToken);
-        await _workflowTraceRepository.UpdateRangeAsync(unhandleTraces, cancellationToken);
-
-        //结束会签
-        var counstersigns = await _workflowCountersignRepository.Queryable()
-            .Where(d => !d.EndTime.HasValue)
-            .ToListAsync(cancellationToken);
-        foreach (var counstersign in counstersigns)
-        {
-            //结束会签
-            counstersign.End(currentStep.Id, currentStep.Code, currentStep.BusinessType,
-                current.UserId, current.UserName,
-                current.OrgId, current.OrgName,
-                current.OrgAreaCode, current.OrgAreaName);
-        }
-
-        await _workflowCountersignRepository.UpdateRangeAsync(counstersigns, cancellationToken);
-
-
-        // //更新实际办理节点信息
-        // workflow.UpdateActualStepWhenHandle(currentStep, current.OrgAreaCode, current.OrgAreaName, current.OrgLevel);
-        //
-        // workflow.UpdateCurrentStepWhenHandle(currentStep, current.OrgAreaCode, current.OrgAreaName, current.OrgLevel);
-
-        if (workflow.Steps.All(d => d.StepType != EStepType.End))
-        {
-            await _workflowDomainService.EndAsync(current, workflow, dto,
-                endStepDefine, currentStep, expiredTime, cancellationToken);
-        }
-    }
-
     /// <summary>
     /// 查询开始流程的下一步待选节点
     /// </summary>
@@ -665,10 +587,13 @@ public class WorkflowApplication : IWorkflowApplication, IScopeDependency
             //ExpiredTime = workflow.ExpiredTime,
             CanStartCountersign = currentStep.CanStartCountersign,
             CurrentStepBusinessType = currentStep.BusinessType,
+            CurrentStepType = currentStep.StepType,
+            CurrentHandlerType = currentStep.HandlerType,
             TimeTypeOptions = EnumExts.GetDescriptions<ETimeType>().ToList(),
             IsMainHandlerShow = workflow.WorkflowDefinition.IsMainHandlerShow,
             StepId = currentStep.Id,
-            CurrentOrgLevel = string.IsNullOrEmpty(currentStep.HandlerOrgId) ? null : currentStep.HandlerOrgId.CalcOrgLevel()
+            CurrentOrgLevel = string.IsNullOrEmpty(currentStep.HandlerOrgId) ? null : currentStep.HandlerOrgId.CalcOrgLevel(),
+            Opinion = currentStep.Opinion,
         };
 
         var currentStepDefine = _workflowDomainService.GetStepDefine(workflow.WorkflowDefinition, currentStep.Code);
@@ -688,22 +613,22 @@ public class WorkflowApplication : IWorkflowApplication, IScopeDependency
             return dto;
         }
 
-        if (currentStep.IsInCountersign())
+        if (workflow.IsInCountersign && currentStep.IsInCountersign())
         {
             if (currentStep.IsCountersignEndStep)
             {
-                // 宜宾需求:会签汇总节点展示会签办理节点办理意见
-                var countersignHandleSteps = workflow.Steps.Where(d =>
-                    d.CountersignId == currentStep.CountersignId &&
-                    d.CountersignPosition is ECountersignPosition.Multi or ECountersignPosition.Single).ToList();
-                var sb = new StringBuilder();
-                foreach (var countersignHandleStep in countersignHandleSteps)
-                {
-                    //sb.AppendLine($"{countersignHandleStep.GetActualHandler()?.GetHandler().Value} : {countersignHandleStep.Opinion}");
-                    sb.AppendLine($"{countersignHandleStep.GetHandler().Value} : {countersignHandleStep.Opinion}");
-                }
-
-                dto.Opinion = sb.ToString();
+                // // 宜宾需求:会签汇总节点展示会签办理节点办理意见
+                // var countersignHandleSteps = workflow.Steps.Where(d =>
+                //     d.CountersignId == currentStep.CountersignId &&
+                //     d.CountersignPosition is ECountersignPosition.Multi or ECountersignPosition.Single).ToList();
+                // var sb = new StringBuilder();
+                // foreach (var countersignHandleStep in countersignHandleSteps)
+                // {
+                //     //sb.AppendLine($"{countersignHandleStep.GetActualHandler()?.GetHandler().Value} : {countersignHandleStep.Opinion}");
+                //     sb.AppendLine($"{countersignHandleStep.GetHandler().Value} : {countersignHandleStep.Opinion}");
+                // }
+                //
+                // dto.Opinion = sb.ToString();
 
                 //当前待办节点为会签汇总节点时:检查是否为顶级会签汇总节点,t:按配置往下走,f:继续往上汇总,不需要重复往下指派
                 if (!currentStep.IsTopCountersignEndStep(workflow.TopCountersignStepId))

+ 1 - 0
src/Hotline.Application/Handlers/FlowEngine/WorkflowEndHandler.cs

@@ -219,6 +219,7 @@ public class WorkflowEndHandler : INotificationHandler<EndWorkflowNotify>
                     orderFlowDto.IsNonPoliceReturn = notification.Dto.External == null ? false : notification.Dto.External.IsPoliceReturn;
                     await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderFiled, orderFlowDto, cancellationToken: cancellationToken);
 
+                    await _orderDomainService.OrderAutomaticPublishAsync(order, cancellationToken);
                     //try
                     //{
                     //    //写入质检  针对受理之后直接结束的工单

+ 4 - 0
src/Hotline.Application/Hotline.Application.csproj

@@ -29,4 +29,8 @@
     <ProjectReference Include="..\XingTang.Sdk\XingTang.Sdk.csproj" />
   </ItemGroup>
 
+  <ItemGroup>
+    <Folder Include="Orders\OrderPublishHandler\" />
+  </ItemGroup>
+
 </Project>

+ 1 - 0
src/Hotline.Application/Mappers/MapperConfigs.cs

@@ -54,6 +54,7 @@ namespace Hotline.Application.Mappers
                 .IgnoreNullValues(true);
 
             config.ForType<TimeLimitSettingInventory, TimeConfig>()
+                .Map(d => d.TimeLimitSettingAttributeId, m => m.Id)
                 .Map(d => d.Count, m => m.TimeValue);
             config.ForType<TimeResult, ExpiredTimeWithConfig>()
                 .Map(d => d.ExpiredTime, x => x.EndTime)

+ 19 - 10
src/Hotline.Application/Mappers/OrderMapperConfigs.cs

@@ -4,6 +4,7 @@ using Hotline.Share.Dtos.Ai;
 using Hotline.Share.Dtos.Order;
 using Hotline.Share.Dtos.Settings;
 using Hotline.Share.Enums.Order;
+using Hotline.Share.Tools;
 using Mapster;
 
 namespace Hotline.Application.Mappers;
@@ -111,7 +112,7 @@ public class OrderMapperConfigs : IRegister
             .Ignore(d => d.AcceptorName)
             .Ignore(d => d.AcceptorStaffNo)
             .Ignore(d => d.ExternalId)
-            .Ignore(d=>d.CreationTime)
+            .Ignore(d => d.CreationTime)
             //.AfterMapping((s, d) =>
             //{
             //    //d.UpdateHandlingStatus(s.IsInCountersign);
@@ -169,15 +170,15 @@ public class OrderMapperConfigs : IRegister
             .IgnoreIf((s, d) => s.VisitDetail == null, d => d.VisitDetail)
             ;
 
-        config.ForType<OrderHandleFlowDto, Workflow>()
-            .Map(d => d.RealHandlerPhone, s => s.RealHandlerPhone)
-            .Map(d => d.RealHandlerName, s => s.RealHandlerName)
-            .Map(d => d.RealCommunicationMode, s => s.RealCommunicationMode)
-            .Map(d => d.RealCommunicationTime, s => s.RealCommunicationTime)
-            .Map(d => d.RealCommunicationAddress, s => s.RealCommunicationAddress)
-            .Map(d => d.RealIsContacted, s => s.RealIsContacted)
-            .Map(d => d.RealContactLocale, s => s.RealContactLocale)
-            .IgnoreNonMapped(true);
+        //config.ForType<OrderHandleFlowDto, Workflow>()
+        //    .Map(d => d.RealHandlerPhone, s => s.RealHandlerPhone)
+        //    .Map(d => d.RealHandlerName, s => s.RealHandlerName)
+        //    .Map(d => d.RealCommunicationMode, s => s.RealCommunicationMode)
+        //    .Map(d => d.RealCommunicationTime, s => s.RealCommunicationTime)
+        //    .Map(d => d.RealCommunicationAddress, s => s.RealCommunicationAddress)
+        //    .Map(d => d.RealIsContacted, s => s.RealIsContacted)
+        //    .Map(d => d.RealContactLocale, s => s.RealContactLocale)
+        //    .IgnoreNonMapped(true);
 
         config.ForType<OrderHandleFlowDto, Order>()
             .Map(src => src.RealCommunicationAddress, dest => dest.RealCommunicationAddress)
@@ -198,12 +199,20 @@ public class OrderMapperConfigs : IRegister
             .IgnoreIf((src, dest) => src.IsOther == null, dest => dest.IsOther)
             .Map(src => src.OtherRemark, dest => dest.OtherRemark)
             .IgnoreIf((src, dest) => string.IsNullOrEmpty(src.OtherRemark), dest => dest.OtherRemark)
+            .Map(src => src.Transpond, dest => dest.Transpond)
+            .IgnoreIf((src, dest) => src.Transpond.HasValue == false, dest => dest.Transpond)
             .Map(src => src.TranspondCityId, dest => dest.TranspondCityId)
             .IgnoreIf((src, dest) => string.IsNullOrEmpty(src.TranspondCityId), dest => dest.TranspondCityId)
             .Map(src => src.TranspondCityName, dest => dest.TranspondCityName)
             .IgnoreIf((src, dest) => string.IsNullOrEmpty(src.TranspondCityName), dest => dest.TranspondCityName)
             .Map(src => src.TranspondCityValue, dest => dest.TranspondCityValue)
             .IgnoreIf((src, dest) => string.IsNullOrEmpty(src.TranspondCityValue), dest => dest.TranspondCityValue)
+            .Map(src => src.IsStepUrgent, dest => dest.IsStepUrgent)
+            .IgnoreIf((src, dest) => src.IsStepUrgent == null, dest => dest.IsStepUrgent)
+            .Map(src => src.IsEvasive, dest => dest.IsEvasive)
+            .IgnoreIf((src, dest) => src.IsEvasive == null, dest => dest.IsEvasive)
+            .Map(src => src.IsInactively, dest => dest.IsInactively)
+            .IgnoreIf((src, dest) => src.IsInactively == null, dest => dest.IsInactively)
             .IgnoreNonMapped(true);
 
         config.ForType<AddOrderComplementDto, OrderComplement>()

+ 18 - 2
src/Hotline.Application/Orders/IOrderApplication.cs

@@ -320,6 +320,22 @@ namespace Hotline.Application.Orders
         /// <returns></returns>
         ISugarQueryable<OrderScreen> OrderScreenList(ScreenListDto dto);
 
-        ISugarQueryable<OrderListOutDto> QueryWaitedForSeat(QueryOrderWaitedDto dto);
-    }
+        /// <summary>
+        /// 待申请列表
+        /// </summary>
+        /// <param name="dto"></param>
+        /// <returns></returns>
+        ISugarQueryable<OrderVisitDetail> MayScreenList(MayScreenListDto dto);
+
+
+		ISugarQueryable<OrderListOutDto> QueryWaitedForSeat(QueryOrderWaitedDto dto);
+
+        /// <summary>
+        /// 受理类型前10
+        /// </summary>
+        /// <param name="dto"></param>
+        /// <returns></returns>
+        Task<(List<SystemDicData> acceptTypes, object items, DataTable data)> AcceptTypeTop10List(ReportPagedRequest dto, bool isExport);
+
+	}
 }

+ 337 - 46
src/Hotline.Application/Orders/OrderApplication.cs

@@ -69,7 +69,7 @@ using XF.Utility.EnumExtensions;
 using Newtonsoft.Json;
 using static NPOI.SS.Format.CellNumberFormatter;
 using System.Linq;
-using DocumentFormat.OpenXml.Bibliography;
+using System.Linq.Dynamic.Core;
 
 namespace Hotline.Application.Orders;
 
@@ -77,6 +77,7 @@ public class OrderApplication : IOrderApplication, IScopeDependency
 {
     private readonly IMediator _mediator;
     private readonly IRepository<TranspondCityRawData> _transpondCityRawDataRepository;
+    private readonly Publisher _publisher;
     private readonly ISessionContextProvider _sessionContextProvider;
     private readonly ISystemDicDataCacheManager _sysDicDataCacheManager;
     private readonly IOptionsSnapshot<AppConfiguration> _appOptions;
@@ -143,6 +144,7 @@ public class OrderApplication : IOrderApplication, IScopeDependency
         IRepository<OrderVisitDetail> orderVisitedDetailRepository,
         IOptionsSnapshot<AppConfiguration> appOptions,
         ISystemDicDataCacheManager sysDicDataCacheManager,
+        Publisher publisher,
         ISessionContextProvider sessionContextProvider,
         IRepository<TranspondCityRawData> transpondCityRawDataRepository,
         IRepository<OrderObserve> orderObserveRepository,
@@ -178,6 +180,7 @@ public class OrderApplication : IOrderApplication, IScopeDependency
         _orderVisitedDetailRepository = orderVisitedDetailRepository;
         _appOptions = appOptions;
         _sysDicDataCacheManager = sysDicDataCacheManager;
+        _publisher = publisher;
         _sessionContextProvider = sessionContextProvider;
         _transpondCityRawDataRepository = transpondCityRawDataRepository;
         _orderObserveRepository = orderObserveRepository;
@@ -216,7 +219,6 @@ public class OrderApplication : IOrderApplication, IScopeDependency
         {
             order.ExpiredTimeProvince = expiredTimeConfig.ExpiredTime;
         }
-        
         //if (string.IsNullOrEmpty(order.WorkflowId))
         //    throw new UserFriendlyException("该工单流程id异常");
         //var workflow = await _workflowDomainService.GetWorkflowAsync(order.WorkflowId, cancellationToken: cancellationToken);
@@ -228,10 +230,6 @@ public class OrderApplication : IOrderApplication, IScopeDependency
         await _workflowDomainService.UpdateUnhandleExpiredTimeAsync(order.WorkflowId, expiredTimeConfig.ExpiredTime, cancellationToken);
 
         await _orderRepository.UpdateAsync(order, cancellationToken);
-
-        //推省上
-        await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderExpiredTimeUpdate,
-           _mapper.Map<OrderDto>(order), cancellationToken: cancellationToken);
     }
 
     // /// <summary>
@@ -329,7 +327,7 @@ public class OrderApplication : IOrderApplication, IScopeDependency
         var orgCode = _sessionContextProvider.SessionContext.OrgId;
         return _orderRepository.Queryable(canView: !IsCenter).Includes(d => d.OrderDelays)
 
-            .WhereIF(orgLevel ==3,d => SqlFunc.Subqueryable<WorkflowStep>()
+            .WhereIF(orgLevel == 3, d => SqlFunc.Subqueryable<WorkflowStep>()
                 .Where(step => step.ExternalId == d.Id && step.Status != EWorkflowStepStatus.Handled &&
                                ((step.FlowAssignType == EFlowAssignType.User && !string.IsNullOrEmpty(step.HandlerId) &&
                                  step.HandlerId == _sessionContextProvider.SessionContext.RequiredUserId) ||
@@ -338,14 +336,46 @@ public class OrderApplication : IOrderApplication, IScopeDependency
                                 (step.FlowAssignType == EFlowAssignType.Role && !string.IsNullOrEmpty(step.RoleId) &&
                                  _sessionContextProvider.SessionContext.Roles.Contains(step.RoleId))))
                 .Any())
-            .WhereIF(orgLevel == 1 || orgLevel == 2, d=> d.ActualHandleOrgCode.StartsWith(orgCode))
+             .WhereIF(orgLevel == 2 || orgLevel == 1, d => SqlFunc.Subqueryable<WorkflowStep>()
+                .Where(step => step.ExternalId == d.Id &&
+                               step.Status != EWorkflowStepStatus.Handled &&
+                               (!string.IsNullOrEmpty(step.HandlerOrgId) && step.HandlerOrgId.StartsWith(orgCode)))
+                .Any())
             .WhereIF(dto.IsProvince.HasValue, d => d.IsProvince == dto.IsProvince)
             .WhereIF(!string.IsNullOrEmpty(dto.No), d => d.No.Contains(dto.No!))
             .WhereIF(!string.IsNullOrEmpty(dto.Title), d => d.Title.Contains(dto.Title!))
             .WhereIF(dto.Delay.HasValue && dto.Delay == 1, d => d.OrderDelays.Any() == true)
             .WhereIF(dto.Delay.HasValue && dto.Delay == 2, d => d.OrderDelays.Any() == false)
+            //&& stTime >= d.ExpiredTime.Value && stTime2 <= d.ExpiredTime.Value
+            //.Where(d => d.ExpiredTime != null &&
+            //         d.Status != EOrderStatus.Filed && d.Status != EOrderStatus.Published && d.Status != EOrderStatus.Visited && stTime >= d.ExpiredTime.Value && stTime2 <= d.ExpiredTime.Value)
             .Where(d => d.Status < EOrderStatus.Filed && dateTime > d.NearlyExpiredTime && dateTime < d.ExpiredTime)
-            .OrderBy(d => d.ExpiredTime);
+            .OrderByIF(dto is { SortField: "no", SortRule: 0 }, x => x.No, OrderByType.Asc) //工单编号升序
+            .OrderByIF(dto is { SortField: "no", SortRule: 1 }, x => x.No, OrderByType.Desc) //工单编号降序
+            .OrderByIF(dto is { SortField: "isProvinceText", SortRule: 0 }, x => x.IsProvince, OrderByType.Asc)//是否省工单升序
+            .OrderByIF(dto is { SortField: "isProvinceText", SortRule: 1 }, x => x.IsProvince, OrderByType.Desc)//是否省工单降序
+            .OrderByIF(dto is { SortField: "currentStepName", SortRule: 0 }, x => x.CurrentStepName, OrderByType.Asc)//当前节点升序
+            .OrderByIF(dto is { SortField: "currentStepName", SortRule: 1 }, x => x.CurrentStepName, OrderByType.Desc)//当前节点降序
+            .OrderByIF(dto is { SortField: "delayText", SortRule: 0 }, x => x.OrderDelays.Count, OrderByType.Asc) //是否延期升序
+            .OrderByIF(dto is { SortField: "delayText", SortRule: 1 }, x => x.OrderDelays.Count, OrderByType.Desc) //是否延期升序
+            .OrderByIF(dto is { SortField: "statusText", SortRule: 0 }, x => x.Status, OrderByType.Asc) //工单状态升序
+            .OrderByIF(dto is { SortField: "statusText", SortRule: 1 }, x => x.Status, OrderByType.Desc) //工单状态升序
+            .OrderByIF(dto is { SortField: "startTime", SortRule: 0 }, x => x.StartTime, OrderByType.Asc) //受理时间升序
+            .OrderByIF(dto is { SortField: "startTime", SortRule: 1 }, x => x.StartTime, OrderByType.Desc) //受理时间降序
+            .OrderByIF(dto is { SortField: "expiredTime", SortRule: 0 }, x => x.ExpiredTime, OrderByType.Asc) //期满时间升序
+            .OrderByIF(dto is { SortField: "expiredTime", SortRule: 1 }, x => x.ExpiredTime, OrderByType.Desc) //期满时间降序
+            .OrderByIF(dto is { SortField: "actualHandleOrgName", SortRule: 0 }, x => x.ActualHandleOrgName, OrderByType.Asc)// 接办部门升序
+            .OrderByIF(dto is { SortField: "actualHandleOrgName", SortRule: 1 }, x => x.ActualHandleOrgName, OrderByType.Desc)// 接办部门降序
+            .OrderByIF(dto is { SortField: "acceptType", SortRule: 0 }, x => x.AcceptTypeCode, OrderByType.Asc) //受理类型升序
+            .OrderByIF(dto is { SortField: "acceptType", SortRule: 1 }, x => x.AcceptTypeCode, OrderByType.Desc) //受理类型降序
+            .OrderByIF(dto is { SortField: "counterSignTypeText", SortRule: 0 }, x => x.CounterSignType, OrderByType.Asc) //是否会签升序
+            .OrderByIF(dto is { SortField: "counterSignTypeText", SortRule: 1 }, x => x.CounterSignType, OrderByType.Desc) //是否会签降序
+            .OrderByIF(dto is { SortField: "orgLevelOneName", SortRule: 0 }, x => x.OrgLevelOneName, OrderByType.Asc) //一级部门升序
+            .OrderByIF(dto is { SortField: "orgLevelOneName", SortRule: 1 }, x => x.OrgLevelOneName, OrderByType.Desc)//一级部门降序
+            .OrderByIF(dto is { SortField: "hotspotName", SortRule: 0 }, x => x.HotspotId, OrderByType.Asc) //热点升序
+            .OrderByIF(dto is { SortField: "hotspotName", SortRule: 1 }, x => x.HotspotId, OrderByType.Desc) //热点降序
+            .OrderByIF(dto is { SortField: "acceptorName", SortRule: 0 }, x => x.AcceptorName, OrderByType.Asc)// 受理人升序
+            .OrderByIF(dto is { SortField: "acceptorName", SortRule: 1 }, x => x.AcceptorName, OrderByType.Desc);// 受理人升序
     }
 
     // /// <summary>
@@ -389,7 +419,7 @@ public class OrderApplication : IOrderApplication, IScopeDependency
         int orgLevel = _sessionContextProvider.SessionContext.OrgLevel;
         var orgCode = _sessionContextProvider.SessionContext.OrgId;
         return _orderRepository.Queryable(canView: false).Includes(d => d.OrderDelays)
-            .WhereIF(orgLevel == 3 , d => SqlFunc.Subqueryable<WorkflowStep>()
+            .WhereIF(orgLevel == 3, d => SqlFunc.Subqueryable<WorkflowStep>()
                 .Where(step => step.ExternalId == d.Id && step.Status != EWorkflowStepStatus.Handled &&
                                ((step.FlowAssignType == EFlowAssignType.User && !string.IsNullOrEmpty(step.HandlerId) &&
                                  step.HandlerId == _sessionContextProvider.SessionContext.RequiredUserId) ||
@@ -398,20 +428,29 @@ public class OrderApplication : IOrderApplication, IScopeDependency
                                 (step.FlowAssignType == EFlowAssignType.Role && !string.IsNullOrEmpty(step.RoleId) &&
                                  _sessionContextProvider.SessionContext.Roles.Contains(step.RoleId))))
                 .Any())
-            .WhereIF(orgLevel == 1 || orgLevel == 2, d => d.ActualHandleOrgCode.StartsWith(orgCode))
+            // .WhereIF(orgLevel == 2 || orgLevel == 1, d =>  d.ActualHandleOrgCode.StartsWith(orgCode))
+            .WhereIF(orgLevel == 2 || orgLevel == 1,  d => SqlFunc.Subqueryable<WorkflowStep>()
+                .Where(step => step.ExternalId == d.Id && 
+                               step.Status != EWorkflowStepStatus.Handled &&
+                               (!string.IsNullOrEmpty(step.HandlerOrgId) && step.HandlerOrgId.StartsWith(orgCode)))
+                .Any())
             .WhereIF(dto.IsProvince.HasValue, d => d.IsProvince == dto.IsProvince)
             //.WhereIF(!string.IsNullOrEmpty(dto.Keyword), d => d.Title.Contains(dto.Keyword!) || d.No.Contains(dto.Keyword!))
             .WhereIF(!string.IsNullOrEmpty(dto.No), x => x.No.Contains(dto.No))
             .WhereIF(!string.IsNullOrEmpty(dto.Title), x => x.Title.Contains(dto.Title!))
             .WhereIF(dto.Delay.HasValue && dto.Delay == 1, d => d.OrderDelays.Any() == true)
             .WhereIF(dto.Delay.HasValue && dto.Delay == 2, d => d.OrderDelays.Any() == false)
-            .Where(d=> d.Status < EOrderStatus.Filed)
+            .Where(d => d.Status < EOrderStatus.Filed)
             .Where(d => d.ExpiredTime != null &&
                         (((d.Status == EOrderStatus.Filed || d.Status == EOrderStatus.Published || d.Status == EOrderStatus.Visited) &&
                           d.FiledTime >= d.ExpiredTime) ||
                          ((d.Status != EOrderStatus.Filed && d.Status != EOrderStatus.Published && d.Status != EOrderStatus.Visited) &&
                           stTime >= d.ExpiredTime.Value)))
-            .OrderBy(x => x.ExpiredTime);
+
+            .OrderByIF(dto is { SortField: "startTime", SortRule: 0 }, x => x.StartTime, OrderByType.Asc) //受理时间升序
+            .OrderByIF(dto is { SortField: "startTime", SortRule: 1 }, x => x.StartTime, OrderByType.Desc) //受理时间降序
+            .OrderByIF(dto is { SortField: "expiredTime", SortRule: 0 }, x => x.ExpiredTime, OrderByType.Asc) //期满时间升序
+            .OrderByIF(dto is { SortField: "expiredTime", SortRule: 1 }, x => x.ExpiredTime, OrderByType.Desc); //期满时间降序
     }
 
     //     /// <summary>
@@ -1011,19 +1050,19 @@ public class OrderApplication : IOrderApplication, IScopeDependency
         var query = _orderRepository.Queryable();
         if (!isCenter)
         {
-            //todo 可优化
-            query.Where(d => SqlFunc.Subqueryable<WorkflowTrace>()
+            query.Where(d => SqlFunc.Subqueryable<WorkflowStep>()
                 .Where(step => step.ExternalId == d.Id &&
                                !string.IsNullOrEmpty(step.HandlerOrgId) &&
-                               step.HandlerOrgId.StartsWith(_sessionContextProvider.SessionContext.RequiredOrgId) &&
-                               step.TraceState != EWorkflowTraceState.StepRemoveByPrevious &&
-                               step.TraceState != EWorkflowTraceState.StepRemoveByRecall).Any());
+                               step.HandlerOrgId.StartsWith(_sessionContextProvider.SessionContext.RequiredOrgId)).Any());
         }
 
-        return query
-             .Includes(x => x.OrderScreens)
-             .Includes(x => x.OrderVisits.Where(m => m.VisitState == EVisitState.Visited).ToList())
-             .Includes(x => x.OrderVisits.Where(m => m.VisitState == EVisitState.Visited).ToList(), ovd => ovd.OrderVisitDetails)
+            query = query.Includes(x => x.OrderScreens);
+            if (!_appOptions.Value.IsYiBin)
+            {
+                query = query.Includes(x => x.OrderVisits.Where(m => m.VisitState == EVisitState.Visited).ToList())
+             .Includes(x => x.OrderVisits.Where(m => m.VisitState == EVisitState.Visited).ToList(), ovd => ovd.OrderVisitDetails);
+            }
+            query = query
              .WhereIF(!string.IsNullOrEmpty(dto.Keyword), d => d.Title.Contains(dto.Keyword!)) //标题
              .WhereIF(!string.IsNullOrEmpty(dto.ProvinceNo), d => d.ProvinceNo == dto.ProvinceNo) //省本地编号
              .WhereIF(!string.IsNullOrEmpty(dto.ReceiveProvinceNo), d => d.ReceiveProvinceNo == dto.ReceiveProvinceNo) //省编号
@@ -1086,13 +1125,53 @@ public class OrderApplication : IOrderApplication, IScopeDependency
                  d.SourceChannelCode == "S12345" && d.IsProvince == true) //省12345
              .WhereIF(!string.IsNullOrEmpty(dto.ContentRetrieval),
                  d => d.Title.Contains(dto.ContentRetrieval) || d.Content.Contains(dto.ContentRetrieval) || d.FileOpinion.Contains(dto.ContentRetrieval) || d.ActualOpinion.Contains(dto.ContentRetrieval))
-             .WhereIF(dto.IsSgin.HasValue && dto.IsSgin == true, d => d.ActualHandleStepAcceptTime != null)
-             .WhereIF(dto.IsSgin.HasValue && dto.IsSgin == false, d => d.ActualHandleStepAcceptTime == null)
+             .WhereIF(dto.IsSgin.HasValue && dto.IsSgin == true, d => d.CurrentStepAcceptTime != null)
+             .WhereIF(dto.IsSgin.HasValue && dto.IsSgin == false, d => d.CurrentStepAcceptTime == null)
              .WhereIF(dto.FiledType is FiledType.CenterFiled, d => d.ProcessType == EProcessType.Zhiban)
              .WhereIF(dto.FiledType is FiledType.OrgFiled, d => d.ProcessType == EProcessType.Jiaoban)
              .WhereIF(!string.IsNullOrEmpty(dto.OrderTagCode), d => d.OrderTagCode == dto.OrderTagCode)
-             .OrderByDescending(d => d.CreationTime);
+             .OrderByIF(string.IsNullOrEmpty(dto.SortField), d => d.CreationTime, OrderByType.Desc)//默认排序时间为创建时间
+             .OrderByIF(dto is { SortField: "no", SortRule: 0 }, d => d.No, OrderByType.Asc) //工单编号升序
+             .OrderByIF(dto is { SortField: "no", SortRule: 1 }, d => d.No, OrderByType.Desc) //工单编号降序
+             .OrderByIF(dto is { SortField: "isProvinceText", SortRule: 0 }, d => d.IsProvince, OrderByType.Asc)//是否省工单升序
+             .OrderByIF(dto is { SortField: "isProvinceText", SortRule: 1 }, d => d.IsProvince, OrderByType.Desc)//是否省工单降序
+             .OrderByIF(dto is { SortField: "reTransactNum", SortRule: 0 }, d => d.ReTransactNum, OrderByType.Asc) //重办次数升序
+             .OrderByIF(dto is { SortField: "reTransactNum", SortRule: 1 }, d => d.ReTransactNum, OrderByType.Desc) //重办次数降序
+             .OrderByIF(dto is { SortField: "isUrgentText",SortRule:0 }, d=>d.IsUrgent,OrderByType.Asc) //是否紧急升序
+             .OrderByIF(dto is { SortField: "isUrgentText",SortRule:1 },d=>d.IsUrgent,OrderByType.Desc) //是否紧急降序
+             .OrderByIF(dto is { SortField: "isSecret",SortRule:0 },d=>d.IsSecret,OrderByType.Asc) //是否紧急升序
+             .OrderByIF(dto is { SortField: "isSecret", SortRule: 1 }, d => d.IsSecret, OrderByType.Desc) //是否紧急降序
+             .OrderByIF(dto is { SortField: "currentStepName", SortRule: 0 }, d => d.CurrentStepName, OrderByType.Asc)//当前节点升序
+             .OrderByIF(dto is { SortField: "currentStepName", SortRule: 1 }, d => d.CurrentStepName, OrderByType.Desc)//当前节点降序
+             .OrderByIF(dto is { SortField: "actualStepAcceptText",SortRule:0 },d=>d.ActualHandleStepAcceptTime.HasValue,OrderByType.Asc) //受理情况升序
+             .OrderByIF(dto is { SortField: "actualStepAcceptText", SortRule: 1 }, d => d.ActualHandleStepAcceptTime.HasValue, OrderByType.Desc) //受理情况降序
+             .OrderByIF(dto is { SortField: "statusText",SortRule:0 },d=>d.Status,OrderByType.Asc) //工单状态升序
+             .OrderByIF(dto is { SortField: "statusText",SortRule:1 },d=>d.Status,OrderByType.Desc) //工单状态降序
+             .OrderByIF(dto is { SortField: "startTime",SortRule:0 },d=>d.StartTime,OrderByType.Asc) //受理时间升序
+             .OrderByIF(dto is { SortField: "startTime",SortRule:1 },d=>d.StartTime,OrderByType.Desc) //受理时间降序
+             .OrderByIF(dto is { SortField: "expiredTime",SortRule:0 },d=>d.ExpiredTime,OrderByType.Asc) //超期时间升序
+             .OrderByIF(dto is { SortField: "expiredTime",SortRule:1 },d=>d.ExpiredTime,OrderByType.Desc) //超期时间降序
+             .OrderByIF(dto is { SortField: "filedTime", SortRule:0 },d=>d.FiledTime,OrderByType.Asc) //办结时间升序
+             .OrderByIF(dto is { SortField: "filedTime", SortRule:1 },d=>d.FiledTime,OrderByType.Desc) //办结时间降序
+             .OrderByIF(dto is { SortField: "orgLevelOneName",SortRule:0 },d=>d.OrgLevelOneName,OrderByType.Asc ) //一级部门升序
+             .OrderByIF(dto is { SortField: "orgLevelOneName",SortRule:1 },d=>d.OrgLevelOneName,OrderByType.Desc ) //一级部门降序
+             .OrderByIF(dto is { SortField: "orgLevelTwoName",SortRule:0 },d=>d.OrgLevelTwoName,OrderByType.Asc) //二级部门升序
+             .OrderByIF(dto is { SortField: "orgLevelTwoName",SortRule:1 },d=>d.OrgLevelTwoName,OrderByType.Desc) //二级部门降序
+             .OrderByIF(dto is { SortField: "actualHandleOrgName",SortRule:0 },d=>d.ActualHandleOrgName,OrderByType.Asc) //接办部门升序
+             .OrderByIF(dto is { SortField: "actualHandleOrgName",SortRule:1 },d=>d.ActualHandleOrgName,OrderByType.Desc) //接办部门降序
+             .OrderByIF(dto is { SortField: "acceptType",SortRule:0 },d=>d.AcceptTypeCode,OrderByType.Asc) //受理类型升序
+             .OrderByIF(dto is { SortField: "acceptType",SortRule:1 },d=>d.AcceptTypeCode,OrderByType.Desc) //受理类型降序
+             .OrderByIF(dto is { SortField: "counterSignTypeText",SortRule:0 },d=>d.CounterSignType,OrderByType.Asc) //是否会签升序
+             .OrderByIF(dto is { SortField: "counterSignTypeText",SortRule:1 },d=>d.CounterSignType,OrderByType.Desc) //是否会签降序
+             .OrderByIF(dto is { SortField: "hotspotSpliceName",SortRule:0 },d=>d.HotspotSpliceName,OrderByType.Asc)//热点分类全称升序
+             .OrderByIF(dto is { SortField: "hotspotSpliceName",SortRule:1 },d=>d.HotspotSpliceName,OrderByType.Desc) //热点分类全称降序
+             .OrderByIF(dto is { SortField: "hotspotName",SortRule:0 },d=>d.HotspotName,OrderByType.Asc) //热点分类升序
+             .OrderByIF(dto is { SortField: "hotspotName",SortRule:1 },d=>d.HotspotName,OrderByType.Desc) //热点分类降序
+             .OrderByIF(dto is { SortField: "acceptorName",SortRule:0 },d=>d.AcceptorName,OrderByType.Asc) //受理人升序
+             .OrderByIF(dto is { SortField: "acceptorName",SortRule:1 },d=>d.AcceptorName,OrderByType.Desc) //受理人降序
+             ;
 
+        return query;
     }
 
     /// <summary>
@@ -1245,8 +1324,8 @@ public class OrderApplication : IOrderApplication, IScopeDependency
         var quer = _orderRepository.Queryable()
             .InnerJoin<SystemOrganize>((x, so) => x.ActualHandleOrgCode == so.Id && so.Level == 1)
             .Where((x, so) => x.CreationTime >= dto.StartTime && x.CreationTime <= dto.EndTime)
-            //.WhereIF(dto.Statuses.Any(), x => dto.Statuses.Contains(x.Status))  //工单状态
-            .WhereIF(dto.QueryType == 1, (x, so) => x.Status >= EOrderStatus.Filed && x.ExpiredTime < x.FiledTime) //业务已办超期
+			//.WhereIF(dto.Statuses.Any(), x => dto.Statuses.Contains(x.Status))  //工单状态
+			.WhereIF(dto.QueryType == 1, (x, so) => x.Status >= EOrderStatus.Filed && x.ExpiredTime < x.FiledTime) //业务已办超期
             .WhereIF(dto.QueryType == 3, (x, so) => x.Status < EOrderStatus.Filed && x.ExpiredTime < SqlFunc.GetDate()) //业务待办超期
             .WhereIF(dto.TypeId != null && dto.TypeId == 1, (x, so) => x.IdentityType == EIdentityType.Citizen)
             .WhereIF(dto.TypeId != null && dto.TypeId == 2, (x, so) => x.IdentityType == EIdentityType.Enterprise)
@@ -1271,7 +1350,8 @@ public class OrderApplication : IOrderApplication, IScopeDependency
                 .Select(x => new { Id = x.ExternalId, WorkflowId = x.ExternalId, HandlerOrgName = x.HandlerOrgName });
 
             quer = _orderRepository.Queryable().InnerJoin(queryCountersign, (o, w) => o.Id == w.Id)
-                .Select((o, w) => new Order { DaysOverdueOrgName = w.HandlerOrgName, Id = o.Id.SelectAll() });
+                .Select((o, w) => new Order { DaysOverdueOrgName = w.HandlerOrgName, Id = o.Id.SelectAll() })
+				.WhereIF(!string.IsNullOrEmpty(dto.No), o => o.No == dto.No);
             //quer = _orderRepository.UnionAll(quer, queryCountersignOrder);
         }
 
@@ -1290,7 +1370,7 @@ public class OrderApplication : IOrderApplication, IScopeDependency
         var quer = _orderRepository.Queryable()
             .InnerJoin<SystemOrganize>((x, so) => x.ActualHandleOrgCode == so.Id && so.Level == 1)
             .Where((x, so) => x.CreationTime >= dto.StartTime && x.CreationTime <= dto.EndTime)
-            .WhereIF(dto.Statuses.Any(), (x, so) => dto.Statuses.Contains(x.Status)) //工单状态
+			.WhereIF(dto.Statuses.Any(), (x, so) => dto.Statuses.Contains(x.Status)) //工单状态
             .WhereIF(dto.ExpiredType is 2, (x, so) => x.OrderDelays.Any(x => x.DelayState == EDelayState.Pass))
             .Where((x, so) =>
                 (x.Status >= EOrderStatus.Filed && x.ExpiredTime < x.FiledTime) ||
@@ -1315,7 +1395,8 @@ public class OrderApplication : IOrderApplication, IScopeDependency
         var queryCountersignOrder = _orderRepository.Queryable().InnerJoin(queryCountersign, (o, w) => o.Id == w.Id)
             .Select((o, w) => new Order { DaysOverdueOrgName = w.HandlerOrgName, Id = o.Id.SelectAll() });
         quer = _orderRepository.UnionAll(quer, queryCountersignOrder).MergeTable()
-            .InnerJoin<SystemOrganize>((x, so) => x.ActualHandleOrgCode == so.Id);
+            .InnerJoin<SystemOrganize>((x, so) => x.ActualHandleOrgCode == so.Id)
+			.WhereIF(!string.IsNullOrEmpty(dto.No), (x, so) => x.No == dto.No);
         return quer;
     }
 
@@ -1329,7 +1410,7 @@ public class OrderApplication : IOrderApplication, IScopeDependency
         return _orderSpecialDetailRepository.Queryable()
             .Includes(x => x.OrderSpecial)
             .WhereIF(!string.IsNullOrEmpty(dto.OrgName), x => x.OrgName.Contains(dto.OrgName!))
-            .Where(x => x.OrderSpecial.SpecialType == ESpecialType.ReTransact)
+            .Where(x => x.OrderSpecial.SpecialType == ESpecialType.ReTransact || x.OrderSpecial.SpecialType == ESpecialType.SendBack )
             .Where(x => x.OrderSpecial.CreationTime >= dto.StartTime)
             .Where(x => x.OrderSpecial.CreationTime <= dto.EndTime)
             .GroupBy(x => new { Time = x.OrderSpecial.CreationTime.ToString("yyyy-MM-dd"), x.OrgId, x.OrgName })
@@ -1359,7 +1440,7 @@ public class OrderApplication : IOrderApplication, IScopeDependency
             .WhereIF(!string.IsNullOrEmpty(dto.OrgName), x => x.OrgName.Contains(dto.OrgName!))
             .WhereIF(!string.IsNullOrEmpty(dto.ErrorName), x => x.ErrorName.Contains(dto.ErrorName!))
             .WhereIF(!string.IsNullOrEmpty(dto.No), x => x.OrderSpecial!.Order!.No!.Contains(dto.No!))
-            .Where(x => x.OrderSpecial.SpecialType == ESpecialType.ReTransact)
+            .Where(x => x.OrderSpecial.SpecialType == ESpecialType.ReTransact || x.OrderSpecial.SpecialType == ESpecialType.SendBack)
             .Where(x => x.OrderSpecial.CreationTime >= dto.StartTime)
             .Where(x => x.OrderSpecial.CreationTime <= dto.EndTime);
     }
@@ -2245,8 +2326,8 @@ public class OrderApplication : IOrderApplication, IScopeDependency
             .Select(x => new OrderScreenAuditVo
             {
                 AuditName = x.HandlerName,
-                AuditNum = SqlFunc.AggregateSum(SqlFunc.IIF(x.TraceState == EWorkflowTraceState.Normal, 1, 0)),
-                AuditBackNum = SqlFunc.AggregateSum(SqlFunc.IIF(x.TraceState == EWorkflowTraceState.StepRemoveByPrevious, 1, 0)),
+                AuditNum = SqlFunc.AggregateSum(SqlFunc.IIF(x.TraceType == EWorkflowTraceType.Normal, 1, 0)),
+                AuditBackNum = SqlFunc.AggregateSum(SqlFunc.IIF(x.TraceType == EWorkflowTraceType.Previous, 1, 0)),
             });
         return query;
     }
@@ -2381,6 +2462,93 @@ public class OrderApplication : IOrderApplication, IScopeDependency
             .OrderByIF(dto is { SortRule: 0, SortField: "creationTime" }, x => x.CreationTime, OrderByType.Asc)
             .OrderByIF(dto is { SortRule: 1, SortField: "creationTime" } || dto.SortRule is null, x => x.CreationTime, OrderByType.Desc);
     }
+
+    public ISugarQueryable<OrderVisitDetail> MayScreenList(MayScreenListDto dto) {
+        var query = _orderVisitedDetailRepository.Queryable(false, true)
+            .Includes(x => x.OrderVisit)
+            .Includes(x => x.OrderVisit, y => y.Order)
+            .Includes(x => x.OrderVisit, y => y.Employee)
+            //.LeftJoin<OrderScreen>((x, s) => x.Id == s.VisitDetailId && s.IsDeleted == false)
+            .Includes(x => x.OrderScreens)
+            .Where(x => x.OrderScreens.Any(s => s.Status == EScreenStatus.SendBack && s.ScreenType == dto.ScreenType && s.SendBackApply == true) ||
+                        x.OrderScreens.Any() == false
+            //|| x.OrderScreens.Any(s => (s.Status != EScreenStatus.SendBack && s.SendBackApply != true)) == false
+            )
+            .WhereIF(dto.ScreenType == EOrderScreenType.Seat, x => x.OrderVisit.Order.IsProvince == false)
+            .WhereIF(dto.ScreenSendBack is 1,
+                x => x.OrderScreens.Any(s => s.Status == EScreenStatus.SendBack && s.ScreenType == dto.ScreenType && s.SendBackApply == true))
+            .WhereIF(dto.ScreenSendBack is 2,
+                x => x.OrderScreens.Any(s => (s.Status != EScreenStatus.SendBack && s.ScreenType == dto.ScreenType && s.SendBackApply != true)) ==
+                     false)
+            .WhereIF(!string.IsNullOrEmpty(dto.No), x => x.OrderVisit.Order!.No!.Contains(dto.No!))
+            .WhereIF(!string.IsNullOrEmpty(dto.Title), x => x.OrderVisit.Order!.Title!.Contains(dto.Title!))
+            .WhereIF(dto.IsProvince.HasValue, x => x.OrderVisit.Order!.IsProvince == dto.IsProvince)
+            .WhereIF(!string.IsNullOrEmpty(dto.AcceptType), x => x.OrderVisit.Order!.AcceptTypeCode! == dto.AcceptType!)
+            .WhereIF(!string.IsNullOrEmpty(dto.HotspotSpliceName),
+                x => x.OrderVisit.Order!.Hotspot.HotSpotFullName!.StartsWith(dto.HotspotSpliceName!))
+            .WhereIF(!string.IsNullOrEmpty(dto.SourceChannel), x => x.OrderVisit.Order!.SourceChannelCode! == dto.SourceChannel!)
+            .WhereIF(!string.IsNullOrEmpty(dto.OrgLevelOneName), x => x.OrderVisit.Order!.OrgLevelOneName!.Contains(dto.OrgLevelOneName!))
+            .WhereIF(!string.IsNullOrEmpty(dto.CurrentHandleOrgName),
+                x => x.OrderVisit.Order!.CurrentHandleOrgName!.Contains(dto.CurrentHandleOrgName!))
+            .WhereIF(!string.IsNullOrEmpty(dto.VisitOrgName), x => x.VisitOrgName!.Contains(dto.VisitOrgName!))
+            .WhereIF(dto.CreationTime.HasValue && dto.EndCreationTime.HasValue,
+                x => x.OrderVisit.Order!.CreationTime >= dto.CreationTime && x.OrderVisit.Order!.CreationTime <= dto.EndCreationTime)
+            .WhereIF(dto.CurrentHandleTime.HasValue && dto.EndCurrentHandleTime.HasValue,
+                x => x.OrderVisit.Order!.CurrentHandleTime >= dto.CurrentHandleTime &&
+                     x.OrderVisit.Order!.ActualHandleTime <= dto.EndCurrentHandleTime)
+            .WhereIF(dto.FiledTime.HasValue && dto.EndFiledTime.HasValue,
+                x => x.OrderVisit.Order!.FiledTime >= dto.FiledTime && x.OrderVisit.Order!.FiledTime <= dto.EndFiledTime)
+            .WhereIF(dto.VisitTime.HasValue && dto.EndVisitTime.HasValue,
+                x => x.OrderVisit.VisitTime >= dto.VisitTime && x.OrderVisit.VisitTime <= dto.EndVisitTime)
+            .WhereIF(dto.IsHomePage.HasValue && dto.IsHomePage == true,
+                x => x.OrderVisit.VisitTime < dto.CreationTimeEnd && x.OrderVisit.VisitTime > dto.CreationTimeStart)
+            //.WhereIF(dto.CounterSignType.HasValue, x => x.OrderVisit.Order!.CounterSignType == dto.CounterSignType)
+            //.WhereIF(!string.IsNullOrEmpty(dto.OrgProcessingResults),
+            //    x => SqlFunc.JsonField(x.OrgProcessingResults, "Key") == dto.OrgProcessingResults)
+            //.WhereIF(!string.IsNullOrEmpty(dto.OrgHandledAttitude),
+            //    x => SqlFunc.JsonListObjectAny(x.OrgHandledAttitude, "Key", dto.OrgHandledAttitude))
+            //.WhereIF(!string.IsNullOrEmpty(dto.OrgNoSatisfiedReason),
+            //    x => SqlFunc.JsonField(x.OrgNoSatisfiedReason, "Key") == dto.OrgNoSatisfiedReason)
+            .Where(x => x.OrderVisit.VisitState == EVisitState.Visited && x.OrderVisit.IsCanHandle);
+		if (_sessionContext.OrgId != null && !_sessionContext.OrgIsCenter)
+		{
+			query.WhereIF(!string.IsNullOrEmpty(dto.Keyword),
+					x => x.OrderVisit.Order.Title.Contains(dto.Keyword!) ||
+						 x.OrderVisit.Order.No.Contains(dto.Keyword!))
+				.Where(x => x.VisitTarget == EVisitTarget.Org && x.VisitOrgCode == _sessionContext.OrgId && (
+					SqlFunc.JsonField(x.OrgProcessingResults, "Key") == "1" ||
+					SqlFunc.JsonField(x.OrgProcessingResults, "Key") == "2" ||
+					SqlFunc.JsonField(x.OrgHandledAttitude, "Key") == "1" ||
+					SqlFunc.JsonField(x.OrgHandledAttitude, "Key") == "2"
+				));
+		}
+		else
+		{
+			query.WhereIF(!string.IsNullOrEmpty(dto.Keyword),
+					x => x.OrderVisit.Order.Title.Contains(dto.Keyword!) ||
+						 x.OrderVisit.Order.No.Contains(dto.Keyword!))
+				.WhereIF(dto.ScreenType == EOrderScreenType.Org, x => x.VisitTarget == EVisitTarget.Org && (
+					SqlFunc.JsonField(x.OrgProcessingResults, "Key") == "1" ||
+					SqlFunc.JsonField(x.OrgProcessingResults, "Key") == "2" ||
+					SqlFunc.JsonField(x.OrgHandledAttitude, "Key") == "1" ||
+					SqlFunc.JsonField(x.OrgHandledAttitude, "Key") == "2"
+				))
+				.WhereIF(dto.ScreenType == EOrderScreenType.Seat,
+					x => x.VisitTarget == EVisitTarget.Seat &&
+						 (x.SeatEvaluate == ESeatEvaluate.VeryNoSatisfied || x.SeatEvaluate == ESeatEvaluate.NoSatisfied))
+				;
+		}
+
+		 return query.OrderByIF(dto is { SortRule: 0, SortField: "order.startTime" }, x => x.OrderVisit.Order.StartTime, OrderByType.Asc)
+			.OrderByIF(dto is { SortRule: 1, SortField: "order.startTime" }, x => x.OrderVisit.Order.StartTime, OrderByType.Desc)
+			.OrderByIF(dto is { SortRule: 0, SortField: "order.actualHandleTime" }, x => x.OrderVisit.Order.ActualHandleTime, OrderByType.Asc)
+			.OrderByIF(dto is { SortRule: 1, SortField: "order.actualHandleTime" }, x => x.OrderVisit.Order.ActualHandleTime, OrderByType.Desc)
+			.OrderByIF(dto is { SortRule: 0, SortField: "order.filedTime" }, x => x.OrderVisit.Order.FiledTime, OrderByType.Asc)
+			.OrderByIF(dto is { SortRule: 1, SortField: "order.filedTime" }, x => x.OrderVisit.Order.FiledTime, OrderByType.Desc)
+			.OrderByIF(dto is { SortRule: 0, SortField: "orderVisit.visitTime" }, x => x.OrderVisit.VisitTime, OrderByType.Asc)
+			.OrderByIF(dto is { SortRule: 1, SortField: "orderVisit.visitTime" }, x => x.OrderVisit.VisitTime, OrderByType.Desc)
+			.OrderByIF(dto.SortRule is null, x => x.CreationTime, OrderByType.Desc);
+	}
     #endregion
 
     #region private
@@ -2477,10 +2645,9 @@ public class OrderApplication : IOrderApplication, IScopeDependency
             }
 
             _mapper.Map(expiredTimeConfig, order);
-            //await _orderRepository.UpdateAsync(order, cancellationToken);
-
-            //特提(撤回至发起)
-            if (!string.IsNullOrEmpty(order.WorkflowId))
+            await _orderRepository.UpdateAsync(order, cancellationToken);
+			//特提(撤回至发起)
+			if (!string.IsNullOrEmpty(order.WorkflowId))
             {
                 //await _workflowDomainService.RecallToStartStepAsync(order.WorkflowId, "省工单重派", current, order.Status >= EOrderStatus.Filed, cancellationToken);
                 var isPaiDan = await _workflowDomainService.RecallToCenterFirstToSendAsync(order.WorkflowId, "省工单重派", order.Status >= EOrderStatus.Filed,
@@ -2494,10 +2661,11 @@ public class OrderApplication : IOrderApplication, IScopeDependency
                 {
                     order.Status = EOrderStatus.WaitForAccept;
                 }
-                //await _orderRepository.UpdateAsync(order, cancellationToken);
-                //处理回访和发布信息
+                await _orderRepository.Updateable().SetColumns(o => new Order{ Status = order.Status }).Where(o => o.Id == order.Id).ExecuteCommandAsync(cancellationToken);
+				//await _orderRepository.UpdateAsync(order, cancellationToken);
+				//处理回访和发布信息
 
-                var publish = await _orderPublishRepository.GetAsync(x => x.OrderId == order.Id);
+				var publish = await _orderPublishRepository.GetAsync(x => x.OrderId == order.Id);
                 if (publish != null)
                 {
                     var publishHistory = _mapper.Map<OrderPublishHistory>(publish);
@@ -2520,7 +2688,6 @@ public class OrderApplication : IOrderApplication, IScopeDependency
                 }
 
             }
-			await _orderRepository.UpdateAsync(order, cancellationToken);
 			//await _workflowDomainService.RecallToStartStepAsync(order.WorkflowId, "省工单重派", current, cancellationToken);
 		}
         return _mapper.Map<AddOrderResponse>(order);
@@ -2763,10 +2930,134 @@ public class OrderApplication : IOrderApplication, IScopeDependency
             .Where(x => x.Source < ESource.MLSQ || x.Source > ESource.WZSC)
             .Where(x => x.Status != EOrderStatus.BackToProvince)
             .WhereIF(dto.TypeCode.HasValue == false, m => m.Status < EOrderStatus.Filed)
-            .OrderBy(d => d.Status)
-            .OrderByIF(dto.IsHandled == true, d => d.StartTime, OrderByType.Desc)
-            .OrderByIF(dto.IsHandled == false, d => new { IsUrgent = d.IsUrgent, CreationTime = d.CreationTime }, OrderByType.Desc)
+             //.OrderBy(d => d.Status)
+             .OrderByIF(dto.IsHandled == true, d => d.StartTime, OrderByType.Desc)
+            .OrderByIF(dto.IsHandled == false, d => new { IsUrgent = d.IsUrgent }, OrderByType.Desc)
+            .OrderByIF(dto.IsHandled == false, d => new { d.Status })
+            .OrderByIF(dto.IsHandled == false && string.IsNullOrEmpty(dto.SortField), d => d.CreationTime, OrderByType.Desc)
+            .OrderByIF(dto is { SortField: "creationTime", SortRule: 0 }, d => d.CreationTime, OrderByType.Asc) //创建时间升序
+            .OrderByIF(dto is { SortField: "creationTime", SortRule: 1 }, d => d.CreationTime, OrderByType.Desc) //创建时间降序
+            .OrderByIF(dto is { SortField: "startTime", SortRule: 0 }, d => d.StartTime, OrderByType.Asc) //受理时间升序
+            .OrderByIF(dto is { SortField: "startTime", SortRule: 1 }, d => d.StartTime, OrderByType.Desc) //受理时间降序
+            .OrderByIF(dto is { SortField: "expiredTime", SortRule: 0 }, d => d.ExpiredTime, OrderByType.Asc) //期满时间升序
+            .OrderByIF(dto is { SortField: "expiredTime", SortRule: 1 }, d => d.ExpiredTime, OrderByType.Desc) //期满时间降序
             .Select<OrderListOutDto>();
     }
-    #endregion
+
+    /// <summary>
+    /// 受理前十
+    /// </summary>
+    /// <param name="dto"></param>
+    /// <param name="isExport"></param>
+    /// <returns></returns>
+	public async Task<(List<SystemDicData> acceptTypes, object items, DataTable data)> AcceptTypeTop10List(ReportPagedRequest dto, bool isExport)
+	{
+
+		var dicList = _systemDicDataRepository.Queryable().Where(x => x.DicTypeCode == "AcceptType").OrderBy(x => x.Sort).MergeTable();
+		var IsCenter = _sessionContext.OrgIsCenter;
+
+		var hotspotList = _hotspotRepository.Queryable().Where(x => SqlFunc.Length(x.Id) == 2)
+			.Select(x => new
+			{
+				HotspotId = x.Id,
+				HotspotName = x.HotSpotFullName,
+			}).MergeTable();
+
+		var orderList = _orderRepository.Queryable()
+			.WhereIF(dto.StartTime.HasValue, x => x.CreationTime >= dto.StartTime)
+			.WhereIF(dto.EndTime.HasValue, x => x.CreationTime <= dto.EndTime)
+			.WhereIF(IsCenter == false, x => x.ActualHandleOrgCode.StartsWith(_sessionContext.RequiredOrgId))
+			.Select(x => new
+			{
+				HotspotId = x.HotspotId.Substring(0, 2),
+				AcceptTypeCode = x.AcceptTypeCode,
+			}).MergeTable();
+
+		var hotListAndOrder = hotspotList.LeftJoin(orderList, (it, o) => it.HotspotId == o.HotspotId)
+			.GroupBy((it, o) => new
+			{
+				it.HotspotId,
+				it.HotspotName,
+				AcceptTypeCode = o.AcceptTypeCode,
+			})
+			.OrderBy((it, o) => it.HotspotId)
+			.Select((it, o) => new
+			{
+				HotspotId = it.HotspotId,
+				HotspotName = it.HotspotName,
+				AcceptTypeCode = o.AcceptTypeCode,
+				Count = SqlFunc.AggregateCount(it.HotspotId)
+			}).MergeTable();
+
+		var returnList = await dicList.LeftJoin(hotListAndOrder, (pp, dd) => pp.DicDataValue == dd.AcceptTypeCode)
+			.GroupBy((pp, dd) => new
+			{
+				HotspotId = dd.HotspotId,
+				HotspotName = dd.HotspotName,
+				AcceptTypeCode = pp.DicDataValue,
+				AcceptType = pp.DicDataName,
+			})
+			.OrderBy((pp, dd) => dd.HotspotId)
+			.Select((pp, dd) => new
+			{
+				HotspotId = dd.HotspotId,
+				HotspotName = dd.HotspotName,
+				AcceptTypeCode = pp.DicDataValue,
+				AcceptType = pp.DicDataName,
+				Count = SqlFunc.AggregateSum(dd.Count)
+			}).ToPivotTableAsync(q => q.AcceptType, q => new { q.HotspotName }, q => q.Sum(x => x.Count));
+		DataColumn totalColumn = new DataColumn("有效受理量", typeof(int));
+		returnList.Columns.Add(totalColumn);
+
+		returnList.Columns["HotspotName"].ColumnName = "省一级热点名称";
+        var titleList = await _systemDicDataRepository.Queryable().Where(x => x.DicTypeCode == "AcceptType").OrderBy(x => x.Sort).ToListAsync();
+		for (int z = 0; z < returnList.Rows.Count; z++)
+		{
+			if (string.IsNullOrEmpty(returnList.Rows[z]["省一级热点名称"].ToString()))
+			{
+				returnList.Rows.Remove(returnList.Rows[z]);
+			}
+			else
+			{
+				int num = 0;
+				var colTotal = returnList.Columns.Count - 1;
+				for (int i = 1; i < colTotal; i++)
+				{
+					num += int.Parse(returnList.Rows[z][i].ToString());
+				}
+				returnList.Rows[z]["有效受理量"] = num;
+			}
+		}
+		returnList.Columns["有效受理量"].SetOrdinal(1);
+		DataTable list = new DataTable();
+		if (returnList.Rows.Count > 0)
+			list = returnList.Select("", "有效受理量 DESC").Take(10).CopyToDataTable();
+		if (isExport)
+		{
+			return (new List<SystemDicData>(), null, list);
+		}
+		var listDy = ToDynamicList(list);
+		return (titleList, listDy, list);
+	}
+
+	public static List<dynamic> ToDynamicList(DataTable dataTable)
+	{
+		var list = new List<dynamic>();
+
+		foreach (DataRow row in dataTable.Rows)
+		{
+			dynamic expando = new ExpandoObject();
+			var expandoDic = (IDictionary<string, object>)expando;
+
+			foreach (DataColumn column in dataTable.Columns)
+			{
+				expandoDic.Add(column.ColumnName, row[column]);
+			}
+
+			list.Add(expando);
+		}
+
+		return list;
+	}
+	#endregion
 }

+ 1 - 1
src/Hotline.Application/Subscribers/DatasharingSubscriber.cs

@@ -814,7 +814,7 @@ namespace Hotline.Application.Subscribers
                     if (order.Status >= EOrderStatus.Filed)
                     {
                         order.ActualOpinion += dto.Opinion;
-                        await _orderRepository.Updateable().UpdateColumns(d => d.ActualOpinion).ExecuteCommandAsync(cancellationToken);
+                        await _orderRepository.Updateable(order).UpdateColumns(d => d.ActualOpinion).ExecuteCommandAsync(cancellationToken);
                         await _workflowDomainService.AppendFileOpinionAsync(order.WorkflowId, dto.Opinion, dto.Files, cancellationToken);
                     }
                     else

+ 14 - 2
src/Hotline.Repository.SqlSugar/Orders/OrderRepository.cs

@@ -1265,7 +1265,13 @@ namespace Hotline.Repository.SqlSugar.Orders
                 .WhereIF(dto.ActualHandleTimeEnd.HasValue, x => x.OrderVisit.Order.ActualHandleTime <= dto.ActualHandleTimeEnd)//办结时间结束
                 .WhereIF(dto.VisitTimeStart.HasValue, x => x.OrderVisit.VisitTime >= dto.VisitTimeStart) //回访时间
                 .WhereIF(dto.VisitTimeEnd.HasValue, x => x.OrderVisit.VisitTime < dto.VisitTimeEnd) //回访时间
-                .OrderByDescending(x => x.OrderVisit.VisitTime)
+                .OrderByIF(string.IsNullOrEmpty(dto.SortField),x=>x.OrderVisit.VisitTime,OrderByType.Desc)
+                .OrderByIF(dto is { SortField: "creationTime", SortRule:0 },x=>x.OrderVisit.Order.CreationTime,OrderByType.Asc) //受理时间升序
+                .OrderByIF(dto is { SortField: "creationTime", SortRule: 1 }, x => x.OrderVisit.Order.CreationTime, OrderByType.Desc) //受理时间降序
+                .OrderByIF(dto is { SortField: "visitTime", SortRule: 0 }, x => x.OrderVisit.VisitTime, OrderByType.Asc) //回访时间升序
+                .OrderByIF(dto is { SortField: "visitTime", SortRule: 1 }, x => x.OrderVisit.VisitTime, OrderByType.Desc) //回访时间降序
+                .OrderByIF(dto is { SortField: "filedTime", SortRule:0 },x=>x.OrderVisit.Order.FiledTime,OrderByType.Asc) //办结时间升序
+                .OrderByIF(dto is { SortField: "filedTime", SortRule: 1 }, x => x.OrderVisit.Order.FiledTime, OrderByType.Desc) //办结时间降序
                 .Select(x => new OrgVisitDetailListResp()
                 {
                     Id = x.OrderVisit.Order.Id,
@@ -1315,7 +1321,13 @@ namespace Hotline.Repository.SqlSugar.Orders
                 .WhereIF(dto.TypeId is 1, x => SqlFunc.JsonField(x.OrgProcessingResults, "Key") == dto.DateValue)
                 .WhereIF(dto.TypeId is 2, x => SqlFunc.JsonField(x.OrgHandledAttitude, "Key") == dto.DateValue)
                 .WhereIF(!string.IsNullOrEmpty(dto.LineNum), x => x.OrderVisit.Order.CallRecord.Gateway == dto.LineNum)
-                .OrderByDescending(x => x.OrderVisit.VisitTime)
+                .OrderByIF(string.IsNullOrEmpty(dto.SortField), x => x.OrderVisit.VisitTime, OrderByType.Desc)
+                .OrderByIF(dto is { SortField: "startTime", SortRule: 0 }, x => x.OrderVisit.Order.StartTime, OrderByType.Asc) //受理时间升序
+                .OrderByIF(dto is { SortField: "startTime", SortRule: 1 }, x => x.OrderVisit.Order.StartTime, OrderByType.Desc) //受理时间降序
+                .OrderByIF(dto is { SortField: "visitTime", SortRule: 0 }, x => x.OrderVisit.VisitTime, OrderByType.Asc) //回访时间升序
+                .OrderByIF(dto is { SortField: "visitTime", SortRule: 1 }, x => x.OrderVisit.VisitTime, OrderByType.Desc) //回访时间降序
+                .OrderByIF(dto is { SortField: "filedTime", SortRule: 0 }, x => x.OrderVisit.Order.FiledTime, OrderByType.Asc) //办结时间升序
+                .OrderByIF(dto is { SortField: "filedTime", SortRule: 1 }, x => x.OrderVisit.Order.FiledTime, OrderByType.Desc) //办结时间降序
                 .Select(x => new OrgVisitDetailListResp
                 {
                     Id = x.OrderVisit.Order.Id,

+ 47 - 0
src/Hotline.Repository.SqlSugar/System/SystemLogRepository.cs

@@ -0,0 +1,47 @@
+using Hotline.Repository.SqlSugar.DataPermissions;
+using Hotline.Settings;
+using Hotline.Share.Tools;
+using SqlSugar;
+using System;
+using System.Collections.Generic;
+using System.Diagnostics;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+using XF.Domain.Dependency;
+
+namespace Hotline.Repository.SqlSugar.System;
+public class SystemLogRepository : BaseRepository<SystemLog>, ISystemLogRepository, IScopeDependency
+{
+    public SystemLogRepository(ISugarUnitOfWork<HotlineDbContext> uow, IDataPermissionFilterBuilder dataPermissionFilterBuilder) : base(uow, dataPermissionFilterBuilder)
+    {
+    }
+
+    public async Task AddAsync(string name, string executeParam = "", string remark = "", string executeUrl = "", int status = 0)
+    {
+        try
+        {
+            var entity = new SystemLog
+            {
+                Name = name,
+                ExecuteParam = executeParam,
+                ExecuteUrl = executeUrl,
+                Remark = remark,
+                Status = status
+            };
+            if (executeUrl.IsNullOrEmpty())
+            {
+                try
+                {
+                    entity.ExecuteUrl = new StackTrace().GetFrame(1).GetMethod().Name;
+                }
+                catch { }
+            }
+            await AddAsync(entity);
+        }
+        catch
+        {
+            // ignore
+        }
+    }
+}

+ 10 - 0
src/Hotline.Share/Dtos/Bi/BiOrderDto.cs

@@ -35,6 +35,16 @@ namespace Hotline.Share.Dtos.Bi
         /// 来电/信人身份0:全部 ,1:市民,2:企业
         /// </summary>
         public int? TypeCode { get; set; }
+
+        /// <summary>
+        /// 排序字段
+        /// </summary>
+        public string? SortField { get; set; }
+
+        /// <summary>
+        /// 排序方式 // 0 升序 1 降序
+        /// </summary>
+        public int? SortRule { get; set; }
     }
 
     public record HighFrequencyCallStatisticsRequest : PagedRequest

+ 1 - 1
src/Hotline.Share/Dtos/CallCenter/TelDto.cs

@@ -66,7 +66,7 @@ namespace Hotline.Share.Dtos.CallCenter
 
         public List<TelGroupDto> Groups { get; set; }
 
-        public string GroupsNo => Groups.IsNullOrEmpty() ? Groups.FirstOrDefault().No : "";
+        public string GroupsNo => Groups.NotNullOrEmpty() ? Groups.FirstOrDefault().No : "";
 
 		#region 添润
 

+ 2 - 1
src/Hotline.Share/Dtos/FlowEngine/NextStepOption.cs

@@ -1,4 +1,5 @@
-using Hotline.Share.Enums.FlowEngine;
+using Hotline.Share.Dtos.FlowEngine.Workflow;
+using Hotline.Share.Enums.FlowEngine;
 
 namespace Hotline.Share.Dtos.FlowEngine;
 

+ 10 - 0
src/Hotline.Share/Dtos/FlowEngine/NextStepsDto.cs

@@ -34,6 +34,16 @@ public class NextStepsDto
     /// </summary>
     public EBusinessType CurrentStepBusinessType { get; set; }
 
+    /// <summary>
+    /// 模板配置节点类型
+    /// </summary>
+    public EStepType CurrentStepType { get; set; }
+
+    /// <summary>
+    /// 办理对象类型
+    /// </summary>
+    public EHandlerType CurrentHandlerType { get; set; }
+    
     /// <summary>
     /// 当前办理节点部门等级(非部门等级办理时无效)
     /// </summary>

+ 23 - 0
src/Hotline.Share/Dtos/Knowledge/KnowledgeDto.cs

@@ -525,4 +525,27 @@ namespace Hotline.Share.Dtos.Knowledge
         /// </summary>
         public string TypeTxt => Type?.GetDescription();
     }
+
+    public class KnowledgeBatchAuditInDto
+    {
+        /// <summary>
+        /// 知识Id集合
+        /// </summary>
+        public string[] KnowledgeIds { get; set; }
+
+        /// <summary>
+        /// 是否通过
+        /// </summary>
+        public bool IsPass { get; set; }
+
+        /// <summary>
+        /// 办理意见
+        /// </summary>
+        public string Opinion { get; set; } = string.Empty;
+
+        /// <summary>
+        /// 是否短信通知
+        /// </summary>
+        public bool IsSms { get; set; }
+    }
 }

+ 18 - 0
src/Hotline.Share/Dtos/Order/Handle/OrderBatchFileDto.cs

@@ -0,0 +1,18 @@
+using Hotline.Share.Dtos.File;
+
+namespace Hotline.Share.Dtos.Order.Handle;
+
+public class OrderBatchFileDto
+{
+    public List<string> OrderIds { get; set; }
+
+    /// <summary>
+    /// 办理意见
+    /// </summary>
+    public string Opinion { get; set; } = string.Empty;
+
+    /// <summary>
+    /// 附件
+    /// </summary>
+    public List<FileDto> Files { get; set; } = new();
+}

+ 8 - 2
src/Hotline.Share/Dtos/Order/OrderBiDto.cs

@@ -2,6 +2,7 @@
 using Hotline.Share.Enums.Order;
 using Hotline.Share.Requests;
 using Microsoft.VisualBasic;
+using System.Data;
 using XF.Utility.EnumExtensions;
 
 namespace Hotline.Share.Dtos.Order
@@ -92,8 +93,13 @@ namespace Hotline.Share.Dtos.Order
 		public int Num { get; set; }
 		public bool Sublevel { get; set; }
 
-		List<HotspotDataLsitVo> Children { get; set; }
-	}
+		public List<HotspotDataLsitVo> Children { get; set; }
+
+		public int ChainNum { get; set; }
+
+		public string ChainRate { get; set; }
+
+    }
 
 
 	public class AcceptTypeTop10Volod {

+ 63 - 13
src/Hotline.Share/Dtos/Order/OrderDto.cs

@@ -133,18 +133,18 @@ namespace Hotline.Share.Dtos.Order
         public double AllDuration { get; set; }
 
 
-        public string AllDurationHour => GetAllDurationHour();
+        public string AllDurationHour { get; set; }
 
-
-        public string GetAllDurationHour()
-        {
-            if (Status >= EOrderStatus.Filed)
-            {
-                return Math.Round(Math.Round((FiledTime - CreationTime).Value.TotalSeconds) / 60 / 60, 2).ToString() + "小时";
-            }
-
-            return "-";
-        }
+        //
+        // public string GetAllDurationHour()
+        // {
+        //     if (Status >= EOrderStatus.Filed && FiledTime.HasValue)
+        //     {
+        //         return Math.Round(Math.Round((FiledTime - CreationTime).Value.TotalSeconds) / 60 / 60, 2).ToString() + "小时";
+        //     }
+        //
+        //     return "-";
+        // }
 
         /// <summary>
         /// 办结时长(秒) 归档时间-受理时间(工单创建时间)
@@ -187,7 +187,7 @@ namespace Hotline.Share.Dtos.Order
         /// 实际办理节点签收时间
         /// </summary>
         public DateTime? ActualHandleStepAcceptTime { get; set; }
-        
+
         public string ActualStepAcceptText => ActualHandleStepAcceptTime.HasValue ? "已签收" : "未签收";
 
         /// <summary>
@@ -269,11 +269,31 @@ namespace Hotline.Share.Dtos.Order
         /// </summary>
         public bool? RealIsContacted { get; set; }
 
+        /// <summary>
+        /// 已与市民沟通
+        /// </summary>
+        public string RealIsContactedTxt => RealIsContacted.HasValue && RealIsContacted.Value ? "已与市民沟通" : string.Empty;
+
         /// <summary>
         /// 已与市民现场沟通
         /// </summary>
         public bool? RealContactLocale { get; set; }
 
+        /// <summary>
+        /// 其它
+        /// </summary>
+        public bool? IsOther { get; set; }
+
+        /// <summary>
+        /// 其它的备注
+        /// </summary>
+        public string? OtherRemark { get; set; }
+
+        /// <summary>
+        /// 已与市民现场沟通
+        /// </summary>
+        public string RealContactLocaleTxt => RealContactLocale.HasValue && RealContactLocale.Value ? "已与市民现场沟通" : string.Empty;
+
         #endregion
 
         #region 当前办理节点信息(指派时赋值)
@@ -530,6 +550,36 @@ namespace Hotline.Share.Dtos.Order
 
         public string IsUrgentText => IsUrgent ? "紧急" : "";
 
+        /// <summary>
+        /// 工单办理时 勾选的 是否紧急
+        /// </summary>
+        public string IsStepUrgenText => IsStepUrgent ? "紧急" : "";
+
+        /// <summary>
+        /// 工单办理时勾选的 是否紧急
+        /// </summary>
+        public bool IsStepUrgent { get; set; }
+
+        /// <summary>
+        /// 是否推诿
+        /// </summary>
+        public bool IsEvasive { get; set; }
+
+        /// <summary>
+        /// 是否推诿
+        /// </summary>
+        public string IsEvasiveTxt => IsEvasive ? "推诿" : "";
+
+        /// <summary>
+        /// 是否不积极
+        /// </summary>
+        public bool IsInactively { get; set; }
+
+        /// <summary>
+        /// 是否不积极
+        /// </summary>
+        public string IsInactivelyTxt => IsInactively ? "不积极" : "";
+
         /// <summary>
         /// 发布范围
         /// </summary>
@@ -1288,7 +1338,7 @@ namespace Hotline.Share.Dtos.Order
         /// <summary>
         /// 备注时间
         /// </summary>
-        public DateTime RemarkTime { get; set; }
+        public DateTime? RemarkTime { get; set; }
     }
 
     public class PublishNearlyExpiredTimeSmsDto

+ 3 - 2
src/Hotline.Share/Dtos/Order/OrderStartFlowDto.cs

@@ -89,9 +89,10 @@ namespace Hotline.Share.Dtos.Order
         public string? OtherRemark { get; set; }
 
         /// <summary>
-        /// 是否紧急
+        /// 禅道_task_59
+        /// 在办理的时候勾选的是否紧急
         /// </summary>
-        public bool? IsUrgent { get; set; }
+        public bool? IsStepUrgent { get; set; }
 
         /// <summary>
         /// 是否推诿

+ 13 - 2
src/Hotline.Share/Dtos/Order/OrderVisitDto.cs

@@ -280,7 +280,13 @@ namespace Hotline.Share.Dtos.Order
         public string UserId { get; set; }
     }
 
-    public class VisitSmsInDto
+    public record VisitPutThroughDto { 
+    
+        public string id { get; set; }
+	}
+
+
+	public class VisitSmsInDto
     {
         [Required]
         public List<string> Ids { get; set; }
@@ -638,7 +644,12 @@ namespace Hotline.Share.Dtos.Order
         /// </summary>
         [Description("短信不满意待回访")]
         SMSUnsatisfied = 41,
-    }
+
+        [Description("未接通")]
+		NoPutThrough =51,
+	}
+
+
 
     public class OrderVisitQuantityOutDto
     {

+ 11 - 1
src/Hotline.Share/Dtos/Order/OrderWaitedDto.cs

@@ -68,7 +68,17 @@ namespace Hotline.Share.Dtos.Order
 		/// </summary>
 		public int? QueryType { get; set; }
 
-	}
+        /// <summary>
+        /// 排序字段
+        /// </summary>
+        public string? SortField { get; set; }
+
+        /// <summary>
+        /// 排序方式 // 0 升序 1 降序
+        /// </summary>
+        public int? SortRule { get; set; }
+
+    }
 
     /// <summary>
     /// 中心待办

+ 10 - 0
src/Hotline.Share/Dtos/Order/PublishedDto.cs

@@ -606,6 +606,16 @@ public record AboutToExpireListDto : PagedKeywordRequest
     /// 0 全部  1 是  2 否
     /// </summary>
     public int? Delay { get; set; }
+
+    /// <summary>
+    /// 排序字段
+    /// </summary>
+    public string? SortField { get; set; } = "expiredTime";
+
+    /// <summary>
+    /// 排序方式 // 0 升序 1 降序
+    /// </summary>
+    public int? SortRule { get; set; } = 0;
 }
 
 public class PublishOrderAllDto

+ 12 - 1
src/Hotline.Share/Dtos/Order/QueryOrderDto.cs

@@ -194,7 +194,18 @@ namespace Hotline.Share.Dtos.Order
         /// 工单标签Code
         /// </summary>
         public string? OrderTagCode { get; set; }
-	}
+
+        /// <summary>
+        /// 排序字段
+        /// </summary>
+        public string? SortField { get; set; }
+
+        /// <summary>
+        /// 排序方式  0 升序 1 降序
+        /// </summary>
+        public int? SortRule { get; set; }
+
+    }
 
 
 	public enum FiledType

+ 5 - 0
src/Hotline.Share/Dtos/Settings/TimeConfig.cs

@@ -80,6 +80,11 @@ namespace Hotline.Share.Dtos.Settings
             TimeText = $"{count}个{timeType.GetDescription()}";
         }
 
+        /// <summary>
+        /// 命中的规则Id
+        /// </summary>
+        public string? TimeLimitSettingAttributeId { get; set; }
+
         public int Count { get; set; }
         public ETimeType TimeType { get; set; }
 

+ 128 - 0
src/Hotline.Share/Dtos/WebPortal/GetCaseReultSendModel.cs

@@ -0,0 +1,128 @@
+using System;
+using System.Collections.Generic;
+using System.ComponentModel.DataAnnotations;
+using System.ComponentModel;
+using System.Linq;
+using System.Text;
+using System.Text.Json.Serialization;
+using System.Threading.Tasks;
+
+namespace Hotline.Share.Dtos.WebPortal
+{
+    public class GetCaseReultSendModel
+    {
+        /// <summary>
+        /// 答复单位
+        /// </summary>
+        [JsonPropertyName("DEPT_NAME")]
+        public string GCRS_DEPT_NAME { get; set; }
+
+        /// <summary>
+        /// 答复时间
+        /// </summary>
+        [JsonPropertyName("FINISH_TIME")]
+        public string? GCRS_FINISH_TIME { get; set; }
+
+        /// <summary>
+        /// 答复人员
+        /// </summary>
+        [JsonPropertyName("FINISH_NAME")]
+        public string GCRS_FINISH_NAME { get; set; }
+
+        /// <summary>
+        /// 办理情况
+        /// </summary>
+        [JsonPropertyName("FINISH_NOTE")]
+        public string GCRS_FINISH_NOTE { get; set; }
+
+        /// <summary>
+        /// 办理操作
+        /// </summary>
+        [JsonPropertyName("FINISH_TYPE")]
+        public string GCRS_FINISH_TYPE { get; set; }
+
+        /// <summary>
+        /// 办理单位
+        /// </summary>
+        [JsonPropertyName("END_DEPT")]
+        public string GCRS_END_DEPT { get; set; }
+
+        /// <summary>
+        /// 办理人员
+        /// </summary>
+        [JsonPropertyName("END_NAME")]
+        public string GCRS_END_NAME { get; set; }
+
+        /// <summary>
+        /// 联系时间
+        /// </summary>
+        [JsonPropertyName("CONCACT_TIME")]
+        public string? GCRS_CONCACT_TIME { get; set; }
+
+        /// <summary>
+        /// 联系方式
+        /// </summary>
+        [JsonPropertyName("CONCACT_TYPE")]
+        public string GCRS_CONCACT_TYPE { get; set; }
+
+        /// <summary>
+        /// 签收时间
+        /// </summary>
+        [JsonPropertyName("SIGN_TIME")]
+        public string? GCRS_SIGN_TIME { get; set; }
+
+        /// <summary>
+        /// 反馈时间
+        /// </summary>
+        [JsonPropertyName("FDBACKTIME")]
+        public string? GCRS_FDBACKTIME { get; set; }
+
+        /// <summary>
+        /// 交办时间
+        /// </summary>
+        [JsonPropertyName("SEND_TIME")]
+        public string? GCRS_SEND_TIME { get; set; }
+
+        /// <summary>
+        /// 反馈意见
+        /// </summary>
+        [JsonPropertyName("FDBACK")]
+        public string GCRS_FDBACK { get; set; }
+
+        /// <summary>
+        /// 办理时长
+        /// </summary>
+        [JsonPropertyName("HANDLETIME_LONG")]
+        public decimal? GCRS_HANDLETIME_LONG { get; set; }
+
+        /// <summary>
+        /// 任务单关联 guid
+        /// </summary>
+        [JsonPropertyName("TROWGUID")]
+        public string GCRS_TROWGUID { get; set; }
+
+        /// <summary>
+        /// 退回类型
+        /// </summary>
+        [JsonPropertyName("BACKTYPE")]
+        public string GCRS_BACKTYPE { get; set; }
+
+        /// <summary>
+        /// 服务工单编号
+        /// </summary>
+        [JsonPropertyName("CASE_SERIAL")]
+        public string GCRS_CASE_SERIAL { get; set; }
+
+        /// <summary>
+        /// 行政区划代码
+        /// </summary>
+        [JsonPropertyName("AREA_CODE")]
+        public string GCRS_AREA_CODE { get; set; }
+
+        /// <summary>
+        /// 材料标识
+        /// </summary>
+        [JsonPropertyName("CLIENG_GUID")]
+        public string GCRS_CLIENG_GUID { get; set; }
+    }
+}

+ 10 - 0
src/Hotline.Share/Requests/DepartmentalProcessingStatisticsDto.cs

@@ -255,6 +255,16 @@ namespace Hotline.Share.Requests
         /// 来电/信人身份0:全部 ,1:市民,2:企业
         /// </summary>
         public int? TypeId { get; set; }
+
+        /// <summary>
+        /// 排序字段
+        /// </summary>
+        public string? SortField { get; set; }
+
+        /// <summary>
+        /// 排序方式 0 升序 1 降序
+        /// </summary>
+        public int? SortRule { get; set; }
     }
 
     public class SelectOrderId

+ 25 - 0
src/Hotline.Share/Requests/PagedKeywordRequest.cs

@@ -502,6 +502,16 @@ public record OrgVisitDetailListReq: PagedKeywordRequest
     /// 来电主体
     /// </summary>
     public int TypeCode { get; set; }
+
+    /// <summary>
+    /// 排序字段
+    /// </summary>
+    public string? SortField { get; set; }
+
+    /// <summary>
+    /// 排序方式 // 0 升序 1 降序
+    /// </summary>
+    public int? SortRule { get; set; }
 }
 
 
@@ -1274,4 +1284,19 @@ public record OrderVisitJudeDetailReq: PagedRequest
 public class OrderVisitJudeDetailRep
 {
 
+}
+
+public class HotspotStatisticsRep {
+	/// <summary>
+	/// 开始时间
+	/// </summary>
+	public DateTime? StartTime { get; set; }
+
+	/// <summary>
+	/// 结束时间
+	/// </summary>
+	public DateTime? EndTime { get; set; }
+
+	public int TypeId { get; set; }
+	public string? HotspotCode { get; set; }
 }

+ 43 - 0
src/Hotline.Share/Tools/HtmlImageValidator.cs

@@ -0,0 +1,43 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Text.RegularExpressions;
+using System.Threading.Tasks;
+
+namespace Hotline.Share.Tools;
+public class HtmlImageValidator
+{
+    private readonly HttpClient _httpClient = new HttpClient();
+
+    public string ValidateAndReplaceImagesAsync(string htmlContent)
+    {
+        string pattern = "<img[^>]+src=\"([^\"]+)\"[^>]*>";
+        var matches = Regex.Matches(htmlContent, pattern);
+
+        foreach (Match match in matches)
+        {
+            string imageUrl = match.Groups[1].Value;
+            if (!IsImageAccessible(imageUrl))
+            {
+                // 将无效链接替换为空白或占位符
+                htmlContent = htmlContent.Replace(imageUrl, "");
+            }
+        }
+
+        return htmlContent;
+    }
+
+    private bool IsImageAccessible(string url)
+    {
+        try
+        {
+            var response = _httpClient.GetAsync(url).GetAwaiter().GetResult();
+            return response.IsSuccessStatusCode;
+        }
+        catch
+        {
+            return false;
+        }
+    }
+}

+ 5 - 0
src/Hotline.Share/Tools/ListExtensions.cs

@@ -8,6 +8,11 @@ namespace Hotline.Share.Tools;
 public static class ListExtensions
 {
     public static bool IsNullOrEmpty<T>(this List<T> value)
+    {
+        return value == null || value.Count == 0;
+    }
+
+    public static bool NotNullOrEmpty<T>(this List<T> value)
     {
         return value != null && value.Count != 0;
     }

+ 5 - 0
src/Hotline/Caching/Interfaces/ISystemSettingCacheManager.cs

@@ -20,5 +20,10 @@ namespace Hotline.Caching.Interfaces
         /// </summary>
         string DefaultVisitEmployeeId { get; }
         int FixedQueryCount { get; }
+
+        /// <summary>
+        /// 自动发布中心直办归档工单
+        /// </summary>
+        bool AutomaticPublishOrder { get; }
     }
 }

+ 32 - 1
src/Hotline/Caching/Services/SystemSettingCacheManager.cs

@@ -39,7 +39,7 @@ namespace Hotline.Caching.Services
         public int EffectiveTimes
             => int.Parse(GetSetting(SettingConstants.EffectiveTimes)?.SettingValue[0]);
 
-        public int ConnectByeTimes 
+        public int ConnectByeTimes
             => int.Parse(GetSetting(SettingConstants.ConnectByeTimes)?.SettingValue[0]);
 
         public int NoConnectByeTimes => int.Parse(GetSetting(SettingConstants.NoConnectByeTimes)?.SettingValue[0]);
@@ -55,5 +55,36 @@ namespace Hotline.Caching.Services
         public string DefaultVisitEmployeeId => GetSetting(SettingConstants.DefaultVisitEmployeeId)?.SettingValue[0].Trim().ToString();
 
         public int FixedQueryCount => int.Parse(GetSetting(SettingConstants.FixedQueryCount)?.SettingValue[0]);
+
+        public bool AutomaticPublishOrder
+        {
+            get
+            {
+
+                try
+                {
+                    var value = GetSetting(SettingConstants.AutomaticPublishOrder)?.SettingValue[0];
+                    if (value == null) return false;
+                    if (value.Trim() == "true")
+                    {
+                        return true;
+                    }
+                }
+                catch (UserFriendlyException e)
+                {
+                    if (e.Message.Contains("无效系统设置"))
+                    {
+                        _systemSettingRepository.AddAsync(new SystemSetting 
+                        {
+                            Code = SettingConstants.AutomaticPublishOrder,
+                            SettingName = "自动发布中心直办归档工单",
+                            SettingValue = ["false"],
+                            Remark = "用于中心直办件归档后默认自动发布, 开启就自动发布.true 或者 false"
+                        });
+                    }
+                }
+                return false;
+            }
+        }
     }
 }

+ 7 - 17
src/Hotline/FlowEngine/Workflows/IWorkflowDomainService.cs

@@ -36,7 +36,7 @@ namespace Hotline.FlowEngine.Workflows
         /// new
         /// </summary>
         Task<List<WorkflowStep>> NextAsync(ISessionContext current, NextWorkflowDto dto,
-            DateTime? expiredTime = null, CancellationToken cancellationToken = default);
+            DateTime? expiredTime = null, bool isAutoFillSummaryOpinion = false, CancellationToken cancellationToken = default);
 
         /// <summary>
         /// 查询工作流
@@ -151,22 +151,6 @@ namespace Hotline.FlowEngine.Workflows
         /// </summary>
         Task<(Kv, IReadOnlyList<Kv>)> GetHandleOrgsAsync(string workflowId, CancellationToken cancellationToken);
 
-        /// <summary>
-        /// 依据配置过滤下一节点
-        /// </summary>
-        List<StepDefine> NextStepDefineFilter(EPathPolicy pathPolicy, List<StepDefine> nextStepDefines);
-
-        // /// <summary>
-        // /// 撤销流程
-        // /// </summary>
-        // Task CancelAsync(CancelDto dto, DateTime? expiredTime, ISessionContext current, CancellationToken cancellationToken);
-
-        ///// <summary>
-        ///// 更新期满时间
-        ///// </summary>
-        //Task UpdateExpiredTimeAsync(Workflow workflow, DateTime expiredTime, string timelimit, int? timelimiteCount,
-        //    ETimeType? timelimitUnit, DateTime nearlyExpiredTime, CancellationToken cancellationToken);
-
         /// <summary>
         /// 新增流程流转记录
         /// </summary>
@@ -302,5 +286,11 @@ namespace Hotline.FlowEngine.Workflows
         /// 追加归档信息(接收ds推送12315归档信息)
         /// </summary>
         Task AppendFileOpinionAsync(string workflowId, string opinion, List<FileDto> files, CancellationToken cancellationToken);
+        
+        /// <summary>
+        /// 跳转至结束节点(无视流程模板配置以及当前办理对象,直接跳至结束节点)
+        /// </summary>
+        Task JumpToEndAsync(ISessionContext current,string workflowId, string opinion, List<FileDto> files, DateTime? expiredTime,
+            EReviewResult reviewResult = EReviewResult.Unknown, CancellationToken cancellationToken = default);
     }
 }

+ 10 - 0
src/Hotline/FlowEngine/Workflows/StepBasicEntity.cs

@@ -371,6 +371,16 @@ public abstract class StepBasicEntity : CreationEntity
 
     #region method
 
+    /// <summary>
+    /// 开启会签
+    /// </summary>
+    public void StartCountersign(string startCountersignId)
+    {
+        IsStartCountersign = true;
+        StartCountersignId = startCountersignId;
+        IsStartedCountersignEnd = false;
+    }
+    
     public bool HasAccepted() => !string.IsNullOrEmpty(AcceptorId);
 
     public void Accept(

+ 38 - 32
src/Hotline/FlowEngine/Workflows/Workflow.cs

@@ -128,40 +128,40 @@ public partial class Workflow : CreationEntity
     [SugarColumn(Length = 8000)]
     public string ActualOpinion { get; set; } = "办理中...";
 
-    /// <summary>
-    /// 真实办理人姓名(手动填写)
-    /// </summary>
-    public string? RealHandlerName { get; set; }
+    ///// <summary>
+    ///// 真实办理人姓名(手动填写)
+    ///// </summary>
+    //public string? RealHandlerName { get; set; }
 
-    /// <summary>
-    /// 真实办理人电话(手动填写)
-    /// </summary>
-    public string? RealHandlerPhone { get; set; }
+    ///// <summary>
+    ///// 真实办理人电话(手动填写)
+    ///// </summary>
+    //public string? RealHandlerPhone { get; set; }
 
-    /// <summary>
-    /// 沟通方式(手动填写)
-    /// </summary>
-    public ERealCommunicationMode? RealCommunicationMode { get; set; }
+    ///// <summary>
+    ///// 沟通方式(手动填写)
+    ///// </summary>
+    //public ERealCommunicationMode? RealCommunicationMode { get; set; }
 
-    /// <summary>
-    /// 沟通时间(手动填写)
-    /// </summary>
-    public DateTime? RealCommunicationTime { get; set; }
+    ///// <summary>
+    ///// 沟通时间(手动填写)
+    ///// </summary>
+    //public DateTime? RealCommunicationTime { get; set; }
 
-    /// <summary>
-    /// 沟通地点(手动填写)
-    /// </summary>
-    public string? RealCommunicationAddress { get; set; }
+    ///// <summary>
+    ///// 沟通地点(手动填写)
+    ///// </summary>
+    //public string? RealCommunicationAddress { get; set; }
 
-    /// <summary>
-    /// 已与市民沟通
-    /// </summary>
-    public bool? RealIsContacted { get; set; }
+    ///// <summary>
+    ///// 已与市民沟通
+    ///// </summary>
+    //public bool? RealIsContacted { get; set; }
 
-    /// <summary>
-    /// 已与市民现场沟通
-    /// </summary>
-    public bool? RealContactLocale { get; set; }
+    ///// <summary>
+    ///// 已与市民现场沟通
+    ///// </summary>
+    //public bool? RealContactLocale { get; set; }
     #endregion
 
     #region 当前办理节点信息(指派时赋值)
@@ -638,7 +638,11 @@ public partial class Workflow
     /// <summary>
     /// 结束流程会签状态(保留topStepId创建节点时判断用,直到下次会签开启时更新)
     /// </summary>
-    public void EndCountersign() => IsInCountersign = false;
+    public void EndCountersign()
+    {
+        IsInCountersign = false;
+        TopCountersignStepId = null;
+    }
 
     /// <summary>
     /// 重置最终办理意见
@@ -887,9 +891,11 @@ public partial class Workflow
 
     public bool IsTopCountersignEndStep(StepBasicEntity step)
     {
-        if (string.IsNullOrEmpty(TopCountersignStepId))
-            throw new UserFriendlyException("该流程当前未开启会签");
-        return step.IsCountersignEndStep && step.CountersignStartStepId == TopCountersignStepId;
+        // if (string.IsNullOrEmpty(TopCountersignStepId))
+        //     throw new UserFriendlyException("该流程当前未开启会签");
+        return step.IsCountersignEndStep 
+               && !string.IsNullOrEmpty(TopCountersignStepId)
+               && step.CountersignStartStepId == TopCountersignStepId;
     }
 
     #endregion

+ 240 - 163
src/Hotline/FlowEngine/Workflows/WorkflowDomainService.cs

@@ -20,6 +20,7 @@ using XF.Domain.Entities;
 using XF.Domain.Exceptions;
 using XF.Domain.Repository;
 using System.Diagnostics;
+using System.Text;
 using Hotline.Share.Dtos.File;
 
 namespace Hotline.FlowEngine.Workflows
@@ -121,7 +122,7 @@ namespace Hotline.FlowEngine.Workflows
 
             //firststeps
             var firstSteps = await CreateNextStepsAsync(workflow, startStep, dto, firstStepDefine,
-                isNextDynamic, flowAssignInfo.FlowAssignType, expiredTime, dto.IsStartCountersign, cancellationToken);
+                isNextDynamic, flowAssignInfo.FlowAssignType, expiredTime, dto.IsStartCountersign, cancellationToken: cancellationToken);
 
             await _workflowStepRepository.UpdateAsync(startStep, cancellationToken);
 
@@ -253,7 +254,7 @@ namespace Hotline.FlowEngine.Workflows
         /// new
         /// </summary>
         public async Task<List<WorkflowStep>> NextAsync(ISessionContext current, NextWorkflowDto dto,
-            DateTime? expiredTime = null, CancellationToken cancellationToken = default)
+            DateTime? expiredTime = null, bool isAutoFillSummaryOpinion = false, CancellationToken cancellationToken = default)
         {
             var workflow = await GetWorkflowAsync(dto.WorkflowId, withDefine: true, withSteps: true,
                 withTraces: true, withCountersigns: true, cancellationToken: cancellationToken);
@@ -337,15 +338,14 @@ namespace Hotline.FlowEngine.Workflows
                 currentStep.FileJson = await _fileRepository.AddFileAsync(
                     dto.Files, workflow.ExternalId, currentStep.Id, cancellationToken);
 
-           var counterSignType = GetCounterSignType(dto.IsStartCountersign, currentStep.BusinessType);
+            var counterSignType = GetCounterSignType(dto.IsStartCountersign, currentStep.BusinessType);
 
             var updateSteps = new List<WorkflowStep> { currentStep };
 
             //结束当前会签流程
             if (currentStep.IsCountersignEndStep)
             {
-                var countersignStartStep =
-                    workflow.Steps.FirstOrDefault(d => d.Id == currentStep.CountersignStartStepId);
+                var countersignStartStep = workflow.Steps.FirstOrDefault(d => d.Id == currentStep.CountersignStartStepId);
                 if (countersignStartStep is null)
                     throw new UserFriendlyException(
                         $"未查询到会签开始step, workflowId: {workflow.Id}, currentStepId: {currentStep.Id}",
@@ -375,26 +375,6 @@ namespace Hotline.FlowEngine.Workflows
 
             await HandleStepAsync(currentStep, workflow, dto, counterSignType, expiredTime, cancellationToken);
 
-
-            var isStartCountersign = currentStep.CountersignPosition switch
-            {
-                ECountersignPosition.None => dto.IsStartCountersign,
-                ECountersignPosition.Multi => !dto.BackToCountersignEnd,
-                ECountersignPosition.Single => !dto.BackToCountersignEnd,
-                ECountersignPosition.End => dto.IsStartCountersign,
-                _ => throw new ArgumentOutOfRangeException()
-            };
-            //创建会签数据
-            if (isStartCountersign)
-            {
-                var exists = workflow.Countersigns.Any(d =>
-                    !d.IsCompleted() && !string.IsNullOrEmpty(current.UserId) && d.StarterId == current.UserId);
-                if (exists)
-                    throw new UserFriendlyException("该用户在当前流程存在未结束会签");
-                await StartCountersignAsync(current, workflow, currentStep, dto, flowAssignInfo.FlowAssignType,
-                    counterSignType, expiredTime, cancellationToken);
-            }
-
             currentStep.IsActualHandled = CheckIsActualHandle(workflow, currentStep, nextStepDefine, dto);
 
             _mapper.Map(dto, workflow);
@@ -405,9 +385,8 @@ namespace Hotline.FlowEngine.Workflows
                 if (!string.IsNullOrEmpty(currentStep.CountersignId))
                 {
                     //会签中正常办理节点,更新会签members办理状态
-                    var countersign =
-                        workflow.Countersigns.FirstOrDefault(d =>
-                            !d.IsCompleted() && d.Id == currentStep.CountersignId);
+                    var countersign = workflow.Countersigns.FirstOrDefault(d =>
+                        !d.IsCompleted() && d.Id == currentStep.CountersignId);
                     if (countersign is not null)
                     {
                         //throw new UserFriendlyException(
@@ -450,17 +429,6 @@ namespace Hotline.FlowEngine.Workflows
                     workflow.EndCountersign();
             }
 
-            // if (workflow.ActualHandleStepId == currentStep.Id)
-            // {
-            //     //更新实际办理节点信息
-            //     //workflow.UpdateActualStepWhenHandle(currentStep, current.OrgAreaCode, current.OrgAreaName, current.OrgLevel);
-            // }
-            //
-            // if (workflow.CurrentStepId == currentStep.Id)
-            // {
-            //     workflow.UpdateCurrentStepWhenHandle(currentStep, current.OrgAreaCode, current.OrgAreaName, current.OrgLevel);
-            // }
-
             //检查是否流转到流程终点
             if (nextStepDefine.StepType is EStepType.End)
             {
@@ -468,10 +436,39 @@ namespace Hotline.FlowEngine.Workflows
                 return new List<WorkflowStep>();
             }
 
+            var isStartCountersign = currentStep.CountersignPosition switch
+            {
+                ECountersignPosition.None => dto.IsStartCountersign,
+                ECountersignPosition.Multi => !dto.BackToCountersignEnd,
+                ECountersignPosition.Single => !dto.BackToCountersignEnd,
+                ECountersignPosition.End => workflow.IsInCountersign || dto.IsStartCountersign,
+                _ => throw new ArgumentOutOfRangeException()
+            };
+            
+            var currentTrace = workflow.Traces.First(d => d.Id == currentStep.Id);
+
+            //创建会签数据
+            if (isStartCountersign)
+            {
+                var exists = workflow.Countersigns.Any(d =>
+                    !d.IsCompleted() && !string.IsNullOrEmpty(current.UserId) && d.StarterId == current.UserId);
+                if (exists)
+                    throw new UserFriendlyException($"该用户在当前流程存在未结束会签, workflowId: {workflow.Id}, userId: {current.UserId}");
+                var countersign = await StartCountersignAsync(current, workflow, currentStep, dto, flowAssignInfo.FlowAssignType,
+                    counterSignType, expiredTime, cancellationToken);
+                currentTrace.StartCountersign(countersign.Id);
+                await _workflowStepRepository.UpdateAsync(currentStep, cancellationToken);
+                await _workflowTraceRepository.UpdateAsync(currentTrace, cancellationToken);
+            }
+
+            //发起会签时记录顶层会签节点
+            if (dto.IsStartCountersign && !workflow.IsInCountersign)
+                workflow.StartCountersign(currentStep.Id, counterSignType);
+
             //创建下一/N个节点(会签汇总节点:会签未全部办理时不创建,最后一个会签办理节点创建会签汇总节点)
             var nextSteps = await CreateNextStepsAsync(workflow, currentStep, dto,
                 nextStepDefine, isNextDynamic, flowAssignInfo.FlowAssignType, expiredTime, dto.IsStartCountersign,
-                cancellationToken);
+                isAutoFillSummaryOpinion, cancellationToken);
 
             // //更新办理对象(nextSteps无元素表示当前节点为会签办理节点且当前会签没有全部办理完成)
             // workflow.UpdateHandlers(current.RequiredUserId, current.RequiredOrgId,
@@ -485,13 +482,6 @@ namespace Hotline.FlowEngine.Workflows
             //更新实际办理节点
             UpdateCurrentStep(workflow, dto, nextStepDefine, nextSteps);
 
-            //发起会签时记录顶层会签节点
-            if (dto.IsStartCountersign && !workflow.IsInCountersign)
-                workflow.StartCountersign(currentStep.Id, counterSignType);
-
-            //更新指派信息
-            //workflow.Assign(flowAssignInfo.FlowAssignType, flowAssignInfo.GetHandlerIds());
-
             //更新会签实际办理对象信息
             if (currentStep.IsActualHandled)
                 workflow.AddCsActualHandler(current.UserId, current.OrgId);
@@ -506,7 +496,6 @@ namespace Hotline.FlowEngine.Workflows
 
             #endregion
 
-            var currentTrace = workflow.Traces.First(d => d.Id == currentStep.Id);
             await _publisher.PublishAsync(
                 new NextStepNotify(workflow, dto, flowAssignInfo, currentTrace, nextStepDefine,
                     current.OrgId, expiredTime.HasValue), PublishStrategy.ParallelWhenAll,
@@ -626,7 +615,7 @@ namespace Hotline.FlowEngine.Workflows
             var currentStep = GetUnHandleStep(workflow.Steps, _sessionContextProvider.SessionContext.RequiredOrgId,
                 _sessionContextProvider.SessionContext.RequiredUserId, _sessionContextProvider.SessionContext.Roles);
             if (currentStep.Status is not EWorkflowStepStatus.WaitForAccept) return null;
-            
+
             //if (currentStep.Handlers.All(d => d.Key != orgId && d.Key != userId)) return null;
 
             if (currentStep.StepType is EStepType.End)
@@ -643,7 +632,7 @@ namespace Hotline.FlowEngine.Workflows
             await _workflowStepRepository.UpdateNav(currentStep)
                 .Include(d => d.WorkflowTrace)
                 .ExecuteCommandAsync();
-            
+
             workflow.ActualHandleStepAcceptTime = currentStep.AcceptTime;
             await _workflowRepository.Updateable(workflow).ExecuteCommandAsync(cancellationToken);
 
@@ -829,7 +818,7 @@ namespace Hotline.FlowEngine.Workflows
             //创建下一/N个节点(会签汇总节点:会签未全部办理时不创建,最后一个会签办理节点创建会签汇总节点)
             var nextSteps = await CreateNextStepsAsync(workflow, currentStep, dto,
                 nextStepDefine, isNextDynamic, flowAssignInfo.FlowAssignType, expiredTime, isStartCountersign,
-                cancellationToken);
+                cancellationToken: cancellationToken);
 
             ////赋值当前节点的下级办理节点
             //if (dto.IsStartCountersign
@@ -1429,7 +1418,7 @@ namespace Hotline.FlowEngine.Workflows
                 throw new UserFriendlyException($"未找到归档节点的前一节点, workflowId: {workflowId}, endStepId: {endStep.Id}");
             prevStep.Opinion += opinion;
 
-            if (files.Any())
+            if (files != null && files.Any())
             {
                 var filejsons = await _fileRepository.AddFileAsync(files, workflow.ExternalId, prevStep.Id, cancellationToken);
                 prevStep.FileJson.AddRange(filejsons);
@@ -1438,6 +1427,83 @@ namespace Hotline.FlowEngine.Workflows
             await _workflowStepRepository.UpdateAsync(prevStep, cancellationToken);
         }
 
+        /// <summary>
+        /// 跳转至结束节点(无视流程模板配置以及当前办理对象,直接跳至结束节点)
+        /// </summary>
+        public async Task JumpToEndAsync(ISessionContext current, string workflowId, string opinion, List<FileDto> files, DateTime? expiredTime,
+            EReviewResult reviewResult = EReviewResult.Unknown, CancellationToken cancellationToken = default)
+        {
+            var workflow = await GetWorkflowAsync(workflowId, withDefine: true, withSteps: true, withTraces: true,
+                withCountersigns: true, cancellationToken: cancellationToken);
+            var endStepDefine = workflow.WorkflowDefinition.FindEndStepDefine();
+            if (endStepDefine is null)
+                throw new UserFriendlyException("未正确配置结束节点");
+
+            var dto = new BasicWorkflowDto
+            {
+                NextStepCode = endStepDefine.Code,
+                NextStepName = endStepDefine.Name,
+                FlowDirection = EFlowDirection.OrgToFile,
+                BusinessType = endStepDefine.BusinessType,
+                ReviewResult = reviewResult,
+                Opinion = opinion,
+                Files = files
+            };
+
+            var unhandleSteps = workflow.Steps
+                .Where(d => d.Status != EWorkflowStepStatus.Handled).ToList();
+            var unhandleTraces = workflow.Traces
+                .Where(d => d.Status != EWorkflowStepStatus.Handled).ToList();
+
+            //get currentStep
+            var currentStep = unhandleSteps.MaxBy(d => d.CreationTime)
+                              ?? workflow.Steps.MaxBy(d => d.CreationTime);
+
+            foreach (var step in unhandleSteps)
+            {
+                await HandleStepAsync(step, workflow, dto, null, null, cancellationToken);
+                if (step.IsStartCountersign)
+                    step.CountersignEnd();
+
+                var trace = unhandleTraces.First(d => d.StepId == step.Id);
+                _mapper.Map(dto, trace);
+                _mapper.Map(step, trace);
+            }
+
+            await _workflowStepRepository.UpdateRangeAsync(unhandleSteps, cancellationToken);
+            await _workflowTraceRepository.UpdateRangeAsync(unhandleTraces, cancellationToken);
+
+            //结束会签
+            var counstersigns = workflow.Countersigns
+                .Where(d => !d.EndTime.HasValue)
+                .ToList();
+            foreach (var counstersign in counstersigns)
+            {
+                //结束会签
+                counstersign.End(currentStep.Id, currentStep.Code, currentStep.BusinessType,
+                    current.UserId, current.UserName,
+                    current.OrgId, current.OrgName,
+                    current.OrgAreaCode, current.OrgAreaName);
+            }
+
+            await _workflowCountersignRepository.UpdateRangeAsync(counstersigns, cancellationToken);
+
+
+            //更新实际办理节点信息
+            if (currentStep.StepType != EStepType.Summary && currentStep.StepType != EStepType.End)
+            {
+                workflow.UpdateActualStepWhenHandle(currentStep, current.OrgAreaCode, current.OrgAreaName, current.OrgLevel);
+                workflow.ActualHandleStepAcceptTime = currentStep.AcceptTime.Value;
+            }
+            //
+            // workflow.UpdateCurrentStepWhenHandle(currentStep, current.OrgAreaCode, current.OrgAreaName, current.OrgLevel);
+
+            if (workflow.Steps.All(d => d.StepType != EStepType.End))
+            {
+                await EndAsync(current, workflow, dto, endStepDefine, currentStep, expiredTime, cancellationToken);
+            }
+        }
+
         /// <summary>
         /// 查找当前会签内所有节点(含start,end)
         /// </summary>
@@ -1781,61 +1847,6 @@ namespace Hotline.FlowEngine.Workflows
             return (new Kv(workflow.ActualHandleOrgCode, workflow.ActualHandleOrgName), items);
         }
 
-        /// <summary>
-        /// 依据配置过滤下一节点
-        /// </summary>
-        public List<StepDefine> NextStepDefineFilter(EPathPolicy pathPolicy, List<StepDefine> nextStepDefines)
-        {
-            switch (pathPolicy)
-            {
-                case EPathPolicy.DirectUpper:
-                    break;
-                case EPathPolicy.DirectUpperCenterIsTop:
-                    var currentOrgLevel = _sessionContextProvider.SessionContext.RequiredOrgId.CalcOrgLevel();
-                    if (currentOrgLevel == 1)
-                    {
-                        nextStepDefines = nextStepDefines.Where(d => d.IsCenter()).ToList();
-                    }
-                    else
-                    {
-                        var upperLevel = (--currentOrgLevel).ToString();
-                        nextStepDefines = nextStepDefines
-                            .Where(d => d.HandlerType is EHandlerType.OrgLevel &&
-                                        d.HandlerTypeItems.Any(x => x.Key == upperLevel))
-                            .ToList();
-                    }
-
-                    break;
-                default:
-                    throw new ArgumentOutOfRangeException();
-            }
-
-            return nextStepDefines;
-        }
-
-        // /// <summary>
-        // /// 撤销流程
-        // /// </summary>
-        // public async Task CancelAsync(CancelDto dto, DateTime? expiredTime, ISessionContext current,
-        //     CancellationToken cancellationToken)
-        // {
-        //     var workflow = await GetWorkflowAsync(dto.WorkflowId, withDefine: true, withSteps: true,
-        //         cancellationToken: cancellationToken);
-        //
-        //     var currentStep = GetUnHandleStep(workflow.Steps, _sessionContextProvider.SessionContext.RequiredOrgId,
-        //         _sessionContextProvider.SessionContext.RequiredUserId, _sessionContextProvider.SessionContext.Roles);
-        //     //var (currentStepBox, currentStep) = GetUnCompleteStep(workflow.Steps, _sessionContextProvider.SessionContext.RequiredOrgId, _sessionContextProvider.SessionContext.RequiredUserId);
-        //
-        //     var endStepDefine = workflow.WorkflowDefinition.FindEndStepDefine();
-        //
-        //     var basicDto = _mapper.Map<BasicWorkflowDto>(dto);
-        //     var endTrace = await EndAsync(workflow, basicDto, endStepDefine, currentStep,
-        //         expiredTime, cancellationToken: cancellationToken);
-        //
-        //     await _publisher.PublishAsync(new CancelWorkflowNotify(workflow), PublishStrategy.ParallelWhenAll,
-        //         cancellationToken);
-        // }
-
         /// <summary>
         /// 新增流程流转记录
         /// </summary>
@@ -1894,12 +1905,6 @@ namespace Hotline.FlowEngine.Workflows
             StepDefine endStepDefine, WorkflowStep currentStep,
             DateTime? expiredTime, CancellationToken cancellationToken)
         {
-            //var endStepHandles = new List<WorkflowStepHandler>
-            //{
-            //    WorkflowStepHandler.Create(workflow.Id, workflow.ExternalId, EFlowAssignType.User,
-            //        current.RequiredUserId, current.UserName, current.RequiredOrgId, current.OrgName)
-            //};
-
             //create endStep
             var endStep = await CreateEndStepAsync(workflow, endStepDefine, currentStep, dto, expiredTime,
                 cancellationToken);
@@ -1914,8 +1919,8 @@ namespace Hotline.FlowEngine.Workflows
             workflow.UpdateCurrentStepWhenHandle(endStep, current.OrgAreaCode, current.OrgAreaName, current.OrgLevel);
             workflow.CurrentStepAcceptTime = endStep.AcceptTime.Value;
 
-            workflow.UpdateActualStepWhenHandle(endStep, current.OrgAreaCode, current.OrgAreaName, current.OrgLevel);
-            workflow.ActualHandleStepAcceptTime = endStep.AcceptTime.Value;
+            // workflow.UpdateActualStepWhenHandle(endStep, current.OrgAreaCode, current.OrgAreaName, current.OrgLevel);
+            // workflow.ActualHandleStepAcceptTime = endStep.AcceptTime.Value;
 
             if (string.IsNullOrEmpty(workflow.OrgLevelOneCode))
                 workflow.UpdateLevelOneOrg(workflow.ActualHandleOrgCode, workflow.ActualHandleOrgName);
@@ -2105,7 +2110,7 @@ namespace Hotline.FlowEngine.Workflows
         /// </summary>
         private async Task<List<WorkflowStep>> CreateNextStepsAsync(Workflow workflow, WorkflowStep currentStep,
             BasicWorkflowDto dto, StepDefine nextStepDefine, bool isNextDynamic, EFlowAssignType flowAssignType,
-            DateTime? expiredTime, bool isStartCountersign, CancellationToken cancellationToken)
+            DateTime? expiredTime, bool isStartCountersign, bool isAutoFillSummaryOpinion = false, CancellationToken cancellationToken = default)
         {
             List<WorkflowStep> nextSteps = new();
             if (currentStep.IsInCountersign())
@@ -2126,7 +2131,7 @@ namespace Hotline.FlowEngine.Workflows
                         {
                             //创建普通节点(根据配置)
                             nextSteps = await CreateConfigStepsAsync(workflow, nextStepDefine, currentStep, dto,
-                                flowAssignType, EWorkflowTraceType.Normal, expiredTime, cancellationToken);
+                                flowAssignType, EWorkflowTraceType.Normal, expiredTime, isAutoFillSummaryOpinion, cancellationToken);
                         }
                     }
                     else
@@ -2139,7 +2144,7 @@ namespace Hotline.FlowEngine.Workflows
                                 throw new UserFriendlyException("未查询到会签节点");
 
                             nextSteps = await CreateCsEndStepsByTargetPrevAsync(workflow, csStartStep, dto, expiredTime,
-                                cancellationToken);
+                                isAutoFillSummaryOpinion, cancellationToken);
                         }
                         else
                         {
@@ -2155,7 +2160,7 @@ namespace Hotline.FlowEngine.Workflows
                     {
                         // check if cs all complete, create next
                         nextSteps = await CreateCsEndStepsByTargetPrevAsync(workflow, currentStep, dto,
-                            expiredTime, cancellationToken);
+                            expiredTime, isAutoFillSummaryOpinion, cancellationToken);
                     }
                     else
                     {
@@ -2181,7 +2186,7 @@ namespace Hotline.FlowEngine.Workflows
             {
                 //创建普通节点(根据配置)
                 nextSteps = await CreateConfigStepsAsync(workflow, nextStepDefine, currentStep, dto, flowAssignType,
-                    EWorkflowTraceType.Normal, expiredTime, cancellationToken);
+                    EWorkflowTraceType.Normal, expiredTime, isAutoFillSummaryOpinion, cancellationToken);
             }
 
             return nextSteps;
@@ -2260,7 +2265,7 @@ namespace Hotline.FlowEngine.Workflows
         /// 根据传入节点的上一节点创建会签汇总节点(汇总传入节点的前一节点)
         /// </summary>
         private async Task<List<WorkflowStep>> CreateCsEndStepsByTargetPrevAsync(Workflow workflow, WorkflowStep step,
-            BasicWorkflowDto dto, DateTime? expiredTime, CancellationToken cancellationToken)
+            BasicWorkflowDto dto, DateTime? expiredTime, bool isAutoFillSummaryOpinion = false, CancellationToken cancellationToken = default)
         {
             var countersignStartStep = workflow.Steps.FirstOrDefault(d => d.Id == step.PrevStepId);
             if (countersignStartStep is null)
@@ -2272,11 +2277,28 @@ namespace Hotline.FlowEngine.Workflows
             if (csInnerSteps.Any(d =>
                     d.Status != EWorkflowStepStatus.Handled || (d.IsStartCountersign && !d.IsStartedCountersignEnd)))
                 return nextSteps;
-            //if (csInnerSteps.All(d => d.Status == EWorkflowStepStatus.Handled))
-            //{
+
+            string? opinion = null;
+            if (isAutoFillSummaryOpinion)
+            {
+                //依据某节点作为根节点,查找最底层办理节点无论普通还是汇总节点
+                var preSteps = GetLastStepsFromRootStep(workflow.Steps, countersignStartStep);
+
+                var sb = new StringBuilder();
+                foreach (var prevStep in preSteps.OrderBy(d => d.HandleTime).ToList())
+                {
+                    sb.AppendLine($"【会签时间】:{prevStep.HandleTime?.ToString("yyyy-MM-dd HH:mm:ss")}");
+                    sb.AppendLine($"【会签人】:{prevStep.HandlerName}");
+                    sb.AppendLine($"【会签结果】:{prevStep.Opinion}");
+                    sb.AppendLine("");
+                }
+
+                opinion = sb.ToString();
+            }
+
             // 创建会签汇总节点
             var countersignEndStep =
-                await CreateCountersignEndStepAsync(workflow, countersignStartStep, dto, expiredTime, cancellationToken);
+                await CreateCountersignEndStepAsync(workflow, countersignStartStep, dto, expiredTime, opinion, cancellationToken);
             nextSteps = new List<WorkflowStep> { countersignEndStep };
 
             //create trace
@@ -2284,14 +2306,46 @@ namespace Hotline.FlowEngine.Workflows
 
             await _publisher.PublishAsync(new CountersignEndAssigned(workflow), PublishStrategy.ParallelWhenAll,
                 cancellationToken);
-            //}
+
 
             return nextSteps;
         }
 
+        /// <summary>
+        /// 以某一节点作为根节点开始查找最底层办理节点(需求:自动填充汇总节点意见)
+        /// </summary>
+        private List<WorkflowStep> GetLastStepsFromRootStep(List<WorkflowStep> steps, WorkflowStep rootStep)
+        {
+            var lastSteps = new List<WorkflowStep>();
+            var nextSteps = steps.Where(d => d.PrevStepId == rootStep.Id).ToList();
+            foreach (var nextStep in nextSteps)
+            {
+                if (nextStep.IsStartCountersign)
+                {
+                    //find last csend
+                    var lastCsEndStep = GetLastCountersignEndStep(steps, nextStep);
+                    if (lastCsEndStep != null)
+                        lastSteps.Add(lastCsEndStep);
+                }
+                else
+                {
+                    lastSteps.Add(nextStep);
+                }
+            }
+
+            return lastSteps;
+        }
+
+        private WorkflowStep? GetLastCountersignEndStep(List<WorkflowStep> steps, WorkflowStep nextStep)
+        {
+            var csEndStep = steps.FirstOrDefault(d => d.IsCountersignEndStep && d.CountersignStartStepId == nextStep.Id);
+            if (csEndStep == null) return null;
+            return !csEndStep.IsStartCountersign ? csEndStep : GetLastCountersignEndStep(steps, csEndStep);
+        }
+
         private async Task<WorkflowStep> CreateCountersignEndStepAsync(
             Workflow workflow, WorkflowStep countersignStartStep,
-            BasicWorkflowDto dto, DateTime? expiredTime,
+            BasicWorkflowDto dto, DateTime? expiredTime, string? opinion = null,
             CancellationToken cancellationToken = default)
         {
             var csEndStep = _mapper.Map<WorkflowStep>(countersignStartStep);
@@ -2311,13 +2365,15 @@ namespace Hotline.FlowEngine.Workflows
             csEndStep.Handlers = countersignStartStep.Handlers
                 .Where(d => d.Key == countersignStartStep.HandlerId || d.Key == countersignStartStep.HandlerOrgId)
                 .ToList();
-            //需求调整:汇总节点指派给发起人部门办理
+            //需求调整:汇总节点指派给发起人部门办理 //todo 待重构
             csEndStep.FlowAssignType = EFlowAssignType.Org;
-            //csEndStep.StepHandlers = stepHandlers;
 
             csEndStep.Reset();
             csEndStep.ResetParameters();
 
+            if (!string.IsNullOrEmpty(opinion))
+                csEndStep.Opinion = opinion;
+
             await _workflowStepRepository.AddAsync(csEndStep, cancellationToken);
             workflow.Steps.Add(csEndStep);
             //await _workflowStepRepository.AddNav(csEndStep)
@@ -2330,7 +2386,8 @@ namespace Hotline.FlowEngine.Workflows
         private bool CheckIsActualHandle(Workflow workflow, WorkflowStep step, StepDefine nextStepDefine,
             BasicWorkflowDto dto)
         {
-            //1. workflow是否为办理类型 2. 非会签:当前是否为普通节点and下一节点是否为汇总 or endStep 3. 会签:当前操作为汇总还是继续往下办理?thk: 汇总以后但未回到top又往下办理的场景,前面实际办理部门也算作办理部门
+            //1. workflow是否为办理类型 2. 非会签:当前是否为普通节点and下一节点是否为汇总 or endStep
+            //3. 会签:当前操作为汇总还是继续往下办理?thk: 汇总以后但未回到top又往下办理的场景,前面实际办理部门也算作办理部门
             if (workflow.FlowType is not EFlowType.Handle) return false;
 
             if (workflow.IsInCountersign)
@@ -2364,15 +2421,21 @@ namespace Hotline.FlowEngine.Workflows
         /// <summary>
         /// 开始会签(创建会签数据,更新currentStep会签数据)
         /// </summary>
-        private async Task StartCountersignAsync(ISessionContext current, Workflow workflow, WorkflowStep startStep,
-            BasicWorkflowDto dto,
-            EFlowAssignType? flowAssignType, ECounterSignType? counterSignType, DateTime? expiredTime,
+        private async Task<WorkflowCountersign> StartCountersignAsync(ISessionContext current, Workflow workflow, WorkflowStep startStep,
+            BasicWorkflowDto dto, EFlowAssignType? flowAssignType, ECounterSignType? counterSignType, DateTime? expiredTime,
             CancellationToken cancellationToken)
         {
+            var countersignId = startStep.CountersignId;
+            if (startStep.IsCountersignEndStep)
+            {
+                var topStartCsStep = GetCsLoopStartStep(workflow, startStep);
+                countersignId = topStartCsStep.CountersignId;
+            }
             var countersign = await CreateCountersignAsync(current, workflow, startStep,
                 dto.NextHandlers.Select(d => new Kv(d.Key, d.Value)).ToList(), flowAssignType,
-                counterSignType, expiredTime, startStep.CountersignId, cancellationToken);
+                counterSignType, expiredTime, countersignId, cancellationToken);
             startStep.StartCountersign(countersign.Id);
+            return countersign;
         }
 
         /// <summary>
@@ -2449,12 +2512,11 @@ namespace Hotline.FlowEngine.Workflows
             newStep.IsOrigin = step.IsOrigin;
             //newStep.ParentId = step.ParentId;
             newStep.Handlers = step.Handlers;
-            //newStep.StepHandlers = _mapper.Map<List<WorkflowStepHandler>>(step.StepHandlers);
             newStep.StartCountersignId = step.StartCountersignId;
             newStep.CountersignId = step.CountersignId;
             newStep.IsStartedCountersignEnd = step.IsStartedCountersignEnd;
 
-            //退回场景:指派给原办理人,其余场景:按照原节点原始指派方式复制
+            //退回场景:指派给原办理人,其余场景:按照原节点原始指派方式复制 //todo 重构为参数传入办理对象
             if (traceType is EWorkflowTraceType.Previous)
             {
                 //newStep.FlowAssignType = EFlowAssignType.User;
@@ -2627,12 +2689,15 @@ namespace Hotline.FlowEngine.Workflows
             trace.TraceType = traceType;
             trace.SendHandleTimes = sendHandleTimes;
 
-            if (step.IsInCountersign())
+            if (workflow.IsInCountersign && step.IsInCountersign())
             {
                 if (step.IsCountersignEndStep)
                 {
-                    var startTrace =
-                        await GetWorkflowTraceAsync(workflow.Id, step.CountersignStartStepId, cancellationToken);
+                    //var startTrace = await GetWorkflowTraceAsync(workflow.Id, step.CountersignStartStepId, cancellationToken);
+                    var startTrace = workflow.Traces.FirstOrDefault(d => d.Id == step.CountersignStartStepId);
+                    if (startTrace == null)
+                        throw new UserFriendlyException(
+                            $"未找到startTrace, workflowId: {workflow.Id}, step.CountersignStartStepId: {step.CountersignStartStepId}");
                     trace.ParentId = startTrace.ParentId;
                 }
                 else
@@ -2647,7 +2712,10 @@ namespace Hotline.FlowEngine.Workflows
                     //    var prevTrace = await GetWorkflowTraceAsync(workflow.Id, step.PrevStepId, cancellationToken);
                     //    trace.ParentId = prevTrace.ParentId;
                     //}
-                    var prevTrace = await GetWorkflowTraceAsync(workflow.Id, step.PrevStepId, cancellationToken);
+                    //var prevTrace = await GetWorkflowTraceAsync(workflow.Id, step.PrevStepId, cancellationToken);
+                    var prevTrace = workflow.Traces.FirstOrDefault(d => d.Id == step.PrevStepId);
+                    if (prevTrace == null)
+                        throw new UserFriendlyException($"未找到prevTrace, workflowId: {workflow.Id}, step.PrevStepId: {step.PrevStepId}");
                     trace.ParentId = prevTrace.Id;
                 }
             }
@@ -2867,7 +2935,8 @@ namespace Hotline.FlowEngine.Workflows
             EFlowAssignType flowAssignType,
             EWorkflowTraceType traceType,
             DateTime? expiredTime,
-            CancellationToken cancellationToken)
+            bool isAutoFillSummaryOpinion = false,
+            CancellationToken cancellationToken = default)
         {
             List<FlowStepHandler> handlers;
             if (stepDefine.HandlerType != EHandlerType.Role && !dto.NextHandlers.Any())
@@ -2885,10 +2954,14 @@ namespace Hotline.FlowEngine.Workflows
                 handlers = dto.NextHandlers;
             }
 
+            string? opinion = null;
+            if (isAutoFillSummaryOpinion && stepDefine.StepType is EStepType.Summary)
+                opinion = prevStep.Opinion;
+
             return await CreateStepsAsync(workflow, stepDefine, prevStep, dto,
                 flowAssignType, handlers, null,
                 EWorkflowStepStatus.WaitForAccept, ECountersignPosition.None,
-                true, traceType, null, expiredTime, cancellationToken);
+                true, traceType, null, expiredTime, opinion, cancellationToken);
         }
 
         private async Task<List<WorkflowStep>> CreateStepsAsync(
@@ -2906,6 +2979,7 @@ namespace Hotline.FlowEngine.Workflows
             EWorkflowTraceType traceType,
             EHandlerType? handlerType = null,
             DateTime? expiredTime = null,
+            string? opinion = null,
             CancellationToken cancellationToken = default
         )
         {
@@ -2915,10 +2989,7 @@ namespace Hotline.FlowEngine.Workflows
                 var isMain = handlers.Count == 1 || (handlers.Count > 1 && handler.Key == dto.NextMainHandler);
                 var step = CreateStep(workflow, stepDefine, prevStep, flowAssignType,
                     handler, dto.NextStepCode, countersignId, stepStatus, csPosition, expiredTime,
-                    dto.NextStepName, isOrigin, isMain, handlerType, dto.BusinessType, dto.FlowDirection);
-
-                //var stepHandler = stepHandlers.First(d => d.GetHandler().Key == handler.Key);
-                //step.StepHandlers = new List<WorkflowStepHandler> { stepHandler };
+                    dto.NextStepName, isOrigin, isMain, handlerType, dto.BusinessType, dto.FlowDirection, opinion);
 
                 steps.Add(step);
             }
@@ -3055,9 +3126,7 @@ namespace Hotline.FlowEngine.Workflows
 
                 //cp会签发起节点变为待办节点
                 //1. create terminal trace 2. 撤回至startStep
-                var newStep = await DuplicateStepWithTraceAsync(workflow, startCountersignStep,
-                    EWorkflowTraceType.Normal,
-                    cancellationToken);
+                var newStep = await DuplicateStepWithTraceAsync(workflow, startCountersignStep, EWorkflowTraceType.Normal, cancellationToken);
 
                 //当topcsStep结束cs时,实际办理节点应该更新为newStep
                 if (startCountersignStep.Id == workflow.TopCountersignStepId)
@@ -3068,7 +3137,9 @@ namespace Hotline.FlowEngine.Workflows
                             UserId = startCountersignStep.HandlerId,
                             Username = startCountersignStep.HandlerName,
                             OrgId = startCountersignStep.HandlerOrgId,
-                            OrgName = startCountersignStep.HandlerOrgName
+                            OrgName = startCountersignStep.HandlerOrgName,
+                            RoleId = startCountersignStep.RoleId,
+                            RoleName = startCountersignStep.RoleName
                         });
 
                     workflow.UpdateCurrentStepWhenAssign(newStep,
@@ -3077,25 +3148,27 @@ namespace Hotline.FlowEngine.Workflows
                             UserId = startCountersignStep.HandlerId,
                             Username = startCountersignStep.HandlerName,
                             OrgId = startCountersignStep.HandlerOrgId,
-                            OrgName = startCountersignStep.HandlerOrgName
+                            OrgName = startCountersignStep.HandlerOrgName,
+                            RoleId = startCountersignStep.RoleId,
+                            RoleName = startCountersignStep.RoleName
                         });
                 }
 
-                //csEndStep又开启了cs,在结束会签时,如果该节点是topcs的end节点, workflow.topcsStep应该更新为前一cs开启stepId
-                if (startCountersignStep.IsTopCountersignEndStep(workflow.TopCountersignStepId))
-                    workflow.TopCountersignStepId = startCountersignStep.CountersignStartStepId;
+                // //csEndStep又开启了cs,在结束会签时,如果该节点是topcs的end节点, workflow.topcsStep应该更新为前一cs开启stepId
+                // if (startCountersignStep.IsTopCountersignEndStep(workflow.TopCountersignStepId))
+                //     workflow.TopCountersignStepId = startCountersignStep.CountersignStartStepId;
 
                 if (workflow.CheckIfCountersignOver())
                     workflow.EndCountersign();
 
-                var removeHandlers = updateSteps.SelectMany(d => d.Handlers).Select(d => d.Key).ToList();
-
-                var handlerObjs = newStep.Handlers.Select(d => new HandlerGroupItem
-                {
-                    GroupId = Guid.NewGuid().ToString(),
-                    Key = d.Key,
-                    Value = d.Value
-                }).ToList();
+                // var removeHandlers = updateSteps.SelectMany(d => d.Handlers).Select(d => d.Key).ToList();
+                //
+                // var handlerObjs = newStep.Handlers.Select(d => new HandlerGroupItem
+                // {
+                //     GroupId = Guid.NewGuid().ToString(),
+                //     Key = d.Key,
+                //     Value = d.Value
+                // }).ToList();
                 //workflow.UpdateHandlers(removeHandlers, newStep.FlowAssignType.Value, handlerObjs);
 
                 await _workflowRepository.UpdateAsync(workflow, cancellationToken);
@@ -3219,7 +3292,8 @@ namespace Hotline.FlowEngine.Workflows
             bool isMainHandler = false,
             EHandlerType? handlerType = null, //动态节点依据动态策略判断
             EBusinessType? businessType = null,
-            EFlowDirection? flowDirection = null
+            EFlowDirection? flowDirection = null,
+            string? opinion = null
         )
         {
             //if (!handlers.Any())
@@ -3243,7 +3317,7 @@ namespace Hotline.FlowEngine.Workflows
             step.IsOrigin = isOrigin;
             step.Name = stepName;
 
-            //新增需求: 部门汇总节点由部门办理//todo 待确认中心由部门处理还是由之前办理人办理
+            //新增需求: 部门汇总节点由部门办理//todo 待确认中心由部门处理还是由之前办理人办理 待重构
             if (step.StepType == EStepType.Summary && step.BusinessType == EBusinessType.Department)
                 step.FlowAssignType = EFlowAssignType.Org;
 
@@ -3256,6 +3330,9 @@ namespace Hotline.FlowEngine.Workflows
             if (businessType.HasValue)
                 step.BusinessType = businessType.Value;
 
+            if (!string.IsNullOrEmpty(opinion))
+                step.Opinion = opinion;
+
             return step;
         }
 

+ 5 - 13
src/Hotline/FlowEngine/Workflows/WorkflowStep.cs

@@ -76,16 +76,6 @@ public class WorkflowStep : StepBasicEntity
     /// </summary>
     public void CountersignEnd() => IsStartedCountersignEnd = true;
 
-    /// <summary>
-    /// 开启会签
-    /// </summary>
-    public void StartCountersign(string startCountersignId)
-    {
-        IsStartCountersign = true;
-        StartCountersignId = startCountersignId;
-        IsStartedCountersignEnd = false;
-    }
-
     //public void CreateCountersignSteps(List<WorkflowStep> nextSteps) =>
     //    CountersignSteps = nextSteps
     //        .Select(d => new CountersignStep { StepId = d.Id })
@@ -169,9 +159,11 @@ public class WorkflowStep : StepBasicEntity
     /// </summary>
     public bool IsTopCountersignEndStep(string? topCountersignStepId)
     {
-        if (string.IsNullOrEmpty(topCountersignStepId))
-            throw new UserFriendlyException($"无效顶级会签节点编号,wfId: {WorkflowId}");
-        return IsCountersignEndStep && CountersignStartStepId == topCountersignStepId;
+        // if (string.IsNullOrEmpty(topCountersignStepId))
+        //     throw new UserFriendlyException($"无效顶级会签节点编号,wfId: {WorkflowId}");
+        return IsCountersignEndStep 
+               && !string.IsNullOrEmpty(topCountersignStepId)
+               && CountersignStartStepId == topCountersignStepId;
     }
 
     //public WorkflowStepHandler?

+ 2 - 2
src/Hotline/Orders/IOrderDomainService.cs

@@ -102,7 +102,7 @@ namespace Hotline.Orders
         /// </summary>
         /// <returns></returns>
         Task SendOverTimeSms(CancellationToken cancellationToken);
-
-
+        Task OrderPublishAsync(Order order, CancellationToken cancellationToken);
+        Task OrderAutomaticPublishAsync(Order order, CancellationToken cancellationToken);
     }
 }

+ 33 - 4
src/Hotline/Orders/Order.cs

@@ -1,5 +1,6 @@
 using Hotline.CallCenter.Calls;
 using Hotline.FlowEngine.Workflows;
+using Hotline.Settings;
 using Hotline.Settings.Hotspots;
 using Hotline.Share.Dtos.File;
 using Hotline.Share.Enums.FlowEngine;
@@ -11,6 +12,7 @@ using System.ComponentModel;
 using XF.Domain.Exceptions;
 using XF.Domain.Extensions;
 using XF.Domain.Repository;
+using XF.Utility.EnumExtensions;
 
 namespace Hotline.Orders
 {
@@ -454,6 +456,9 @@ namespace Hotline.Orders
 
         [SugarColumn(ColumnDescription = "全流程工作日时长")]
         public double? AllDurationWorkday { get; set; }
+        
+        
+        public string AllDurationHour { get; set; }
 
         /// <summary>
         /// 办理时间限制(如:24小时、7个工作日)
@@ -964,10 +969,18 @@ namespace Hotline.Orders
 
         /// <summary>
         /// 是否紧急
+        /// 在创单的时候勾选的是否紧急
         /// </summary>
         [SugarColumn(DefaultValue = "f", ColumnDescription = "是否紧急")]
         public bool IsUrgent { get; set; }
 
+        /// <summary>
+        /// 是否紧急
+        /// 在办理的时候勾选的是否紧急
+        /// </summary>
+        [SugarColumn(DefaultValue = "f", ColumnDescription = "办理勾选是否紧急")]
+        public bool IsStepUrgent { get; set; }
+
         /// <summary>
         /// 是否推诿
         /// </summary>
@@ -1039,7 +1052,8 @@ namespace Hotline.Orders
 		/// </summary>
 		[SugarColumn(ColumnDescription = "话务提醒是否转办")]
 		public bool? IsForwarded { get; set; }
-	}
+
+    }
 
     public partial class Order
     {
@@ -1351,10 +1365,25 @@ namespace Hotline.Orders
             }
         }
 
-        #endregion
-    }
+        /// <summary>
+        /// 自动发布条件检测
+        /// </summary>
+        /// <returns></returns>
+        /// <exception cref="NotImplementedException"></exception>
+        public string AutoPublishCheck()
+        {
+            if (CounterSignType != null)
+                return "会签工单自动发布失败;";
+            if (IsProvince)
+                return "省工单自动发布失败;";
+            if (Status != EOrderStatus.Filed)
+                return "工单状态非 已归档, " + Status.GetDescription();
+            return "";
+        }
+    #endregion
+}
 
-    public class UnsignedOrder
+public class UnsignedOrder
     {
         public Order Order { get; set; }
 

+ 207 - 2
src/Hotline/Orders/OrderDomainService.cs

@@ -24,6 +24,15 @@ using SqlSugar;
 using Hotline.Push.Notifies;
 using Hotline.Share.Enums.Push;
 using MediatR;
+using Hotline.Authentications;
+using Hotline.ContingencyManagement.Notifies;
+using Hotline.EventBus;
+using Hotline.Share.Enums.FlowEngine;
+using Mapster;
+using Microsoft.AspNetCore.Builder.Extensions;
+using Hotline.Configurations;
+using Microsoft.Extensions.Options;
+using Hotline.Share.Tools;
 
 namespace Hotline.Orders;
 
@@ -38,16 +47,20 @@ public class OrderDomainService : IOrderDomainService, IScopeDependency
     private readonly ITypedCache<CacheOrderNO> _cacheOrderNo;
     private readonly ISessionContext _sessionContext;
     private readonly ICapPublisher _capPublisher;
+    private readonly IRepository<OrderVisitDetail> _orderVisitDetailRepository;
     private readonly IMapper _mapper;
     private readonly ILogger<OrderDomainService> _logger;
     private readonly IFileRepository _fileRepository;
     private readonly IRepository<Scheduling> _schedulingRepository;
+    private readonly Publisher _publisher;
+    private readonly IOptionsSnapshot<AppConfiguration> _appOptions;
     private readonly IRepository<User> _userRepository;
     private readonly ISystemSettingCacheManager _systemSettingCacheManager;
     private readonly IWorkflowDomainService _workflowDomainService;
     private readonly IRepository<Hotspot> _hotspotRepository;
     private readonly IMediator _mediator;
-
+    private readonly IRepository<WorkflowStep> _workflowStepRepository;
+    private readonly ISystemLogRepository _systemLogRepository;
 
     public OrderDomainService(
         IOrderRepository orderRepository,
@@ -68,7 +81,12 @@ public class OrderDomainService : IOrderDomainService, IScopeDependency
         IRepository<Scheduling> schedulingRepository,
         IWorkflowDomainService workflowDomainService,
         IRepository<Hotspot> hotspotRepository,
-        IMediator mediator)
+        IMediator mediator,
+        Publisher publisher,
+        IRepository<OrderVisitDetail> orderVisitDetailRepository,
+        IRepository<WorkflowStep> workflowStepRepository,
+        ISystemLogRepository systemLogRepository,
+        IOptionsSnapshot<AppConfiguration> appOptions)
     {
         _orderRepository = orderRepository;
         _orderRedoRepository = orderRedoRepository;
@@ -88,6 +106,191 @@ public class OrderDomainService : IOrderDomainService, IScopeDependency
         _workflowDomainService = workflowDomainService;
         _hotspotRepository = hotspotRepository;
         _mediator = mediator;
+        _publisher = publisher;
+        _orderVisitDetailRepository = orderVisitDetailRepository;
+        _workflowStepRepository = workflowStepRepository;
+        _systemLogRepository = systemLogRepository;
+        _appOptions = appOptions;
+    }
+
+    /// <summary>
+    /// 归档后自动发布
+    /// </summary>
+    /// <param name="order"></param>
+    /// <param name="cancellationToken"></param>
+    /// <returns></returns>
+    public async Task OrderAutomaticPublishAsync(Order order, CancellationToken cancellationToken)
+    {
+        var name = "中心直办件归档后默认自动发布";
+        try
+        {
+            if (_systemSettingCacheManager.AutomaticPublishOrder == false)
+            {
+                return;
+            }
+            var checkResult = order.AutoPublishCheck();
+            if (checkResult.NotNullOrEmpty())
+            {
+                await _systemLogRepository.AddAsync(name, order.Id, checkResult);
+                return;
+            }
+
+            if (order.ProcessType != EProcessType.Zhiban)
+            {
+                await _systemLogRepository.AddAsync(name, order.Id, "非中心直办件");
+                return;
+            }
+
+            await OrderPublishAsync(order, cancellationToken);
+        }
+        catch (Exception e)
+        {
+            await _systemLogRepository.AddAsync(name, order.Id, $"系统异常: {e.Message}");
+        }
+    }
+
+    public async Task OrderPublishAsync(Order order, CancellationToken cancellationToken)
+    {
+        OrderPublish orderPublish = new()
+        {
+            OrderId = order.Id,
+            No = order.No,
+            PublishState = false, //当前写死为false
+            ArrangeTitle = order.Title,
+            ArrangeContent = order.Content,
+            ArrangeOpinion = order.FileOpinion,
+            ProPublishState = false,
+            FeedBackPhone = order.Contact,
+            CreatorName = _sessionContext.UserName
+        };
+        await _orderPublishRepository.AddAsync(orderPublish, cancellationToken);
+        order.Publish(orderPublish.PublishState);
+        await _orderRepository.UpdateAsync(order, cancellationToken);
+        //推省上
+        var publishPublishOrder = orderPublish.Adapt<PublishPublishOrderDto>();
+        publishPublishOrder.Order = order.Adapt<OrderDto>();
+        //查询实际办理附件
+        if (!string.IsNullOrEmpty(order.ActualHandleStepId)) 
+        {
+            var actualHandleStep = await _workflowStepRepository.GetAsync(order.ActualHandleStepId, cancellationToken);
+            publishPublishOrder.FileJsons = actualHandleStep?.FileJson;
+        }
+
+        await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderPublishOrder, publishPublishOrder, cancellationToken: cancellationToken);
+
+        //推应急管理局
+        //是否开启
+        var isOpenContingencyManagement =
+            _systemSettingCacheManager.GetSetting(SettingConstants.IsOpenContingencyManagement)?.SettingValue[0];
+        if (isOpenContingencyManagement == "true")
+            await _publisher.PublishAsync(new ContingencyManagementNotify(order, order.Title, order.Content, order.ActualOpinion),
+                PublishStrategy.ParallelWhenAll, cancellationToken);
+
+        var orderVisit = new OrderVisit
+        {
+            No = order.No,
+            OrderId = order.Id,
+            VisitState = EVisitState.WaitForVisit,
+            PublishTime = DateTime.Now,
+            IsCanHandle = true,
+            EmployeeId = _sessionContext.RequiredUserId
+        };
+        if (!order.IsProvince)
+        {
+            orderVisit.EmployeeId = _sessionContext.RequiredUserId;
+        }
+
+        if (_appOptions.Value.IsZiGong)
+        {
+            orderVisit.EmployeeId = string.Empty;
+        }
+
+        if (order is { ProcessType: EProcessType.Zhiban, CounterSignType: null } && !order.IsProvince)
+        {
+            orderVisit.VisitState = EVisitState.Visited;
+            orderVisit.VisitTime = DateTime.Now;
+            orderVisit.VisitType = EVisitType.OtherVisit;
+            orderVisit.NowEvaluate = new Kv() { Key = "4", Value = "满意" };
+            if (_appOptions.Value.IsZiGong)
+            {
+                // 根据禅道 自贡需求 Id_361, 第一条, 3小条需求;
+                // 直办件归档后自动回访量需统计在“胡玲”的默认回访量中;
+                orderVisit.EmployeeId = _systemSettingCacheManager.DefaultVisitEmployeeId;
+            }
+        }
+
+        if (order.CounterSignType != ECounterSignType.Center)
+        {
+            orderVisit.IsCanAiVisit = true;
+        }
+
+        string visitId = await _orderVisitRepository.AddAsync(orderVisit);
+
+        //新增回访信息
+        var visitedDetail = new List<OrderVisitDetail>();
+
+        var seatDetail = new OrderVisitDetail();
+        seatDetail.VisitId = visitId;
+        seatDetail.VisitTarget = EVisitTarget.Seat;
+
+
+        var orgDetail = new OrderVisitDetail();
+        orgDetail.VisitId = visitId;
+        orgDetail.VisitOrgCode = order.ActualHandleOrgCode;
+        orgDetail.VisitOrgName = order.ActualHandleOrgName;
+        orgDetail.VisitTarget = EVisitTarget.Org;
+        if (order is { ProcessType: EProcessType.Zhiban, CounterSignType: null, IsProvince: false })
+        {
+            var satisfy = new Kv() { Key = "4", Value = "满意" };
+            orgDetail.OrgProcessingResults = satisfy;
+            //orgDetail.OrgHandledAttitude = satisfy;
+        }
+
+        visitedDetail.Add(orgDetail);
+
+        if (order is { ProcessType: EProcessType.Zhiban, CounterSignType: null })
+        {
+            seatDetail.VoiceEvaluate = EVoiceEvaluate.Satisfied;
+            seatDetail.SeatEvaluate = ESeatEvaluate.Satisfied;
+            order.Visited("4", "满意");
+            order.Status = EOrderStatus.Visited;
+            await _orderRepository.UpdateAsync(order, cancellationToken);
+        }
+
+        visitedDetail.Add(seatDetail);
+        await _orderVisitDetailRepository.AddRangeAsync(visitedDetail, cancellationToken);
+
+        if (order.IsProvince == false && orderVisit.VisitState == EVisitState.Visited)
+        {
+            //推省上
+            await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderVisited,
+                new PublishVisitDto()
+                {
+                    Order = order.Adapt<OrderDto>(),
+                    No = orderVisit.No,
+                    VisitType = orderVisit.VisitType,
+                    VisitName = orderVisit.CreatorName,
+                    VisitTime = orderVisit.VisitTime,
+                    VisitRemark = orderVisit.NowEvaluate?.Value,
+                    AreaCode = order.AreaCode!,
+                    SubjectResultSatifyCode = orderVisit.NowEvaluate?.Key,
+                    FirstSatisfactionCode = orderVisit.NowEvaluate?.Key,
+                    ClientGuid = ""
+                }, cancellationToken: cancellationToken);
+        }
+
+        //推门户
+        await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderVisitedWeb, new PublishVisitAllDto()
+        {
+            Id = orderVisit.Id,
+            Order = order.Adapt<OrderDto>(),
+            OrderVisitDetails = orderVisit.OrderVisitDetails.Adapt<List<VisitDetailDto>>(),
+            VisitName = _sessionContext.UserName,
+            VisitTime = orderVisit.VisitTime,
+            VisitType = orderVisit.VisitType,
+            VisitState = orderVisit.VisitState,
+            PublishTime = orderVisit.PublishTime,
+        }, cancellationToken: cancellationToken);
     }
 
     public async Task<Order> GetOrderAsync(string? orderId, bool withHotspot = false, bool withAcceptor = false,
@@ -328,6 +531,7 @@ public class OrderDomainService : IOrderDomainService, IScopeDependency
         }
     }
     #endregion
+
     #region  工单校验- 交通类工单
 
     /// <summary>
@@ -541,6 +745,7 @@ public class OrderDomainService : IOrderDomainService, IScopeDependency
         return $"{today:yyyyMMdd}{count:000000}";
     }
 
+
     #endregion
 }
 

+ 21 - 0
src/Hotline/Settings/ISystemLogRepository.cs

@@ -0,0 +1,21 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+using XF.Domain.Repository;
+
+namespace Hotline.Settings;
+public interface ISystemLogRepository : IRepository<SystemLog>
+{
+    /// <summary>
+    /// 添加操作日志
+    /// </summary>
+    /// <param name="name">方法</param>
+    /// <param name="executeParam">入参</param>
+    /// <param name="executeUrl">地址</param>
+    /// <param name="remark">备注</param>
+    /// <param name="status">状态(0失败 1成功)</param>
+    /// <returns></returns>
+    Task AddAsync(string name, string executeParam = "", string remark = "", string executeUrl = "", int status = 0);
+}

+ 10 - 0
src/Hotline/Settings/SettingConstants.cs

@@ -565,5 +565,15 @@ namespace Hotline.Settings
         /// 上传文件格式显示
         /// </summary>
         public const string FileExt = "FileExt";
+
+        /// <summary>
+        /// 自动发布中心直办归档工单
+        /// </summary>
+        public const string AutomaticPublishOrder = "AutomaticPublishOrder";
+
+        /// <summary>
+        /// 是否开启自动填写办理意见至汇总节点
+        /// </summary>
+        public const string IsAutoFillSummaryOpinion = "IsAutoFillSummaryOpinion";
     }
 }

+ 54 - 0
src/Hotline/dataview.md

@@ -458,3 +458,57 @@ where visittemp."CreationTime">='2024-08-29' and visitdetailtemp."VisitTarget"=1
 on aaa."SugarNav_Id" = ccc."OrderId"
 
 
+### 重置工单回访部门信息
+update order_visit_detail set 
+"VisitOrgCode"= hhh."HandlerOrgId","VisitOrgName"=hhh."HandlerOrgName"
+from (
+select * from (
+select ppp."OrderId",ooo."Id" "VisitDetailId" from(
+select "Id","OrderId" from order_visit where "VisitState" <> 50 and "OrderId" in(
+select "Id" from "order" WHERE "FiledTime">'2024-10-30' and "ActualHandleOrgCode" <> '001' and "Status">300 and "CounterSignType" is null)) ppp
+left join order_visit_detail ooo on ppp."Id"=ooo."VisitId"
+WHERE ooo."VisitTarget"=20) qqq
+left join 
+(select aaa.* from workflow_step aaa
+left join (
+select "ExternalId","max"("HandleTime") maxTime from workflow_step where "WorkflowId" in(
+select "WorkflowId" from "order" WHERE "FiledTime">'2024-10-30' and "ActualHandleOrgCode"<>'001' and "Status">=300 and "CounterSignType" is null)
+and  "StepType" not in (2,3) AND "IsOrigin"=TRUE group  by "ExternalId") bbb on aaa."ExternalId"=bbb."ExternalId" 
+where aaa."HandleTime"=bbb.maxTime) www on qqq."OrderId"= www."ExternalId") hhh
+where order_visit_detail."Id" = hhh."VisitDetailId" and hhh."OrderId"='08dcf7df-964a-436c-8f39-bc96ff54522a'
+
+
+### 重置工单实际办理部门和接办信息
+update  "order" uuu
+set
+"ActualHandleOrgName"=ccc."HandlerOrgName","ActualHandleOrgCode"=ccc."HandlerOrgId","CurrentHandleOrgId"=ccc."HandlerOrgId","CurrentHandlerName"=ccc."HandlerOrgName"
+from (
+select aaa."ExternalId",aaa."HandlerOrgId",aaa."HandlerOrgName" from workflow_step aaa
+left join (
+select "ExternalId","max"("HandleTime") maxTime from workflow_step where "WorkflowId" in(
+select "WorkflowId" from "order" WHERE "FiledTime">'2024-10-30' and "ActualHandleOrgCode"<>'001' and "Status">=300 and "CounterSignType" is null)
+and  "StepType" not in (2,3) AND "IsOrigin"=TRUE group  by "ExternalId") bbb on aaa."ExternalId"=bbb."ExternalId" 
+where aaa."HandleTime"=bbb.maxTime) ccc where uuu."Id"=ccc."ExternalId" and uuu."Id"='08dcf7df-964a-436c-8f39-bc96ff54522a'
+
+
+
+### 查询智能回访超过两次的回访工单
+update order_visit_detail set "SeatEvaluate"=4,"VisitContent"='市民未接听电话。'
+where "VisitId" in(
+select "Id" from order_visit WHERE "OrderId" in(
+select "OrderId" from ai_order_visit_detail where "CreationTime">'2024-10-29'
+and "CallTimes">0 and "AiSeatEvaluate" is null and "AiIsContact" is null and "AiVolved" is  NULL  and "AiOrgProcessingResults" is null
+group by "OrderId") and "VisitState" not in(50,30)) and "VisitTarget"=10;
+
+
+update order_visit_detail set "OrgProcessingResults" ='{"Key":"6","Value":"未接通"}',"VisitContent"='市民未接听电话。',"IsContact"=TRUE,"Volved"=true
+where "VisitId" in(
+select "Id" from order_visit WHERE "OrderId" in(
+select "OrderId" from ai_order_visit_detail where "CreationTime">'2024-10-29'
+and "CallTimes">0 and "AiSeatEvaluate" is null and "AiIsContact" is null and "AiVolved" is  NULL  and "AiOrgProcessingResults" is null
+group by "OrderId") and "VisitState" not in(50,30)) and "VisitTarget"=20
+
+
+
+
+

+ 2 - 2
src/XF.Domain/Extensions/StringExtensions.cs

@@ -59,8 +59,8 @@ namespace XF.Domain.Extensions
         {
             if (string.IsNullOrEmpty(str) || str.Length == 0)
                 return 0;
-            ASCIIEncoding ascii = new ASCIIEncoding();
-            int tempLen = 0;
+            var ascii = new ASCIIEncoding();
+            var tempLen = 0;
             byte[] s = ascii.GetBytes(str);
             for (int i = 0; i < s.Length; i++)
             {