Order.cs 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664
  1. using Hotline.FlowEngine.Workflows;
  2. using Hotline.Settings.Hotspots;
  3. using Hotline.Share.Dtos;
  4. using Hotline.Share.Enums.FlowEngine;
  5. using Hotline.Share.Enums.Order;
  6. using Hotline.Share.Enums.Settings;
  7. using Hotline.Users;
  8. using SqlSugar;
  9. using XF.Domain.Exceptions;
  10. using XF.Domain.Extensions;
  11. using XF.Domain.Repository;
  12. namespace Hotline.Orders
  13. {
  14. /// <summary>
  15. /// 工单
  16. /// </summary>
  17. public partial class Order : PositionWorkflowEntity
  18. {
  19. #region 来电信息
  20. /// <summary>
  21. /// 来源渠道(电话、网站、APP等)
  22. /// </summary>
  23. [SugarColumn(DefaultValue = "其他")]
  24. public string SourceChannel { get; set; }
  25. [SugarColumn(DefaultValue = "QT")]
  26. public string SourceChannelCode { get; set; }
  27. /// <summary>
  28. /// 渠道为电话时,此字段存在(Call.Id)
  29. /// </summary>
  30. [SugarColumn(IsNullable = true)]
  31. public string? CallId { get; set; }
  32. /// <summary>
  33. /// 来电归属地
  34. /// </summary>
  35. [SugarColumn(IsNullable = true)]
  36. public string? CallAddress { get; set; }
  37. /// <summary>
  38. /// 来电号码
  39. /// </summary>
  40. [SugarColumn(IsNullable = true)]
  41. public string? FromPhone { get; set; }
  42. /// <summary>
  43. /// 转接号码(转接来源)
  44. /// </summary>
  45. [SugarColumn(IsNullable = true)]
  46. public string? TransferPhone { get; set; }
  47. /// <summary>
  48. /// 来电/信人姓名
  49. /// </summary>
  50. public string FromName { get; set; }
  51. /// <summary>
  52. /// 来电/信人性别
  53. /// </summary>
  54. public EGender FromGender { get; set; }
  55. /// <summary>
  56. /// 来电/信人身份
  57. /// </summary>
  58. public EIdentityType IdentityType { get; set; }
  59. /// <summary>
  60. /// 证件类型
  61. /// </summary>
  62. [SugarColumn(IsNullable = true)]
  63. public string? LicenceTypeCode { get; set; }
  64. [SugarColumn(IsNullable = true)]
  65. public string? LicenceType { get; set; }
  66. /// <summary>
  67. /// 证件号码
  68. /// </summary>
  69. [SugarColumn(IsNullable = true)]
  70. public string? LicenceNo { get; set; }
  71. /// <summary>
  72. /// 年龄段
  73. /// </summary>
  74. [SugarColumn(IsNullable = true)]
  75. public string? AgeRangeCode { get; set; }
  76. [SugarColumn(IsNullable = true)]
  77. public string? AgeRange { get; set; }
  78. /// <summary>
  79. /// 联系电话
  80. /// </summary>
  81. [SugarColumn(ColumnDescription = "联系电话", IsNullable = true)]
  82. public string? Contact { get; set; }
  83. [SugarColumn(ColumnDescription = "联系电话脱敏", IsNullable = true)]
  84. public string? ContactMask { get; set; }
  85. /// <summary>
  86. /// 是否接受短信,勾选校验手机号
  87. /// </summary>
  88. public bool AcceptSms { get; set; }
  89. /// <summary>
  90. /// 是否已发送短信
  91. /// </summary>
  92. public bool SmsSended { get; set; }
  93. /// <summary>
  94. /// 是否保密
  95. /// </summary>
  96. [SugarColumn(DefaultValue = "t")]
  97. public bool IsSecret { get; set; }
  98. /// <summary>
  99. /// 工作单位(当“来电/信人身份”为“企业”时必填,其他情况非必填)
  100. /// </summary>
  101. [SugarColumn(ColumnDescription = "工作单位", IsNullable = true)]
  102. public string? Company { get; set; }
  103. #endregion
  104. #region 诉求信息
  105. /// <summary>
  106. /// 工单类型
  107. /// </summary>
  108. public EOrderType OrderType { get; set; }
  109. /// <summary>
  110. /// 受理类型
  111. /// </summary>
  112. public EAcceptType AcceptType { get; set; }
  113. /// <summary>
  114. /// 紧急程度
  115. /// </summary>
  116. public EEmergencyLevel EmergencyLevel { get; set; } = EEmergencyLevel.Normal;
  117. public string Title { get; set; }
  118. #region 热点
  119. /// <summary>
  120. /// 热点
  121. /// </summary>
  122. public string HotspotId { get; set; }
  123. public string HotspotName { get; set; }
  124. public string HotspotSpliceName { get; set; }
  125. /// <summary>
  126. /// 外部数据(为前端提供级联功能)
  127. /// </summary>
  128. [SugarColumn(IsNullable = true)]
  129. public string? HotspotExternal { get; set; }
  130. #endregion
  131. /// <summary>
  132. /// 事发时间
  133. /// </summary>
  134. public DateTime? IncidentTime { get; set; }
  135. /// <summary>
  136. /// 重复工单Id
  137. /// </summary>
  138. [SugarColumn(ColumnDescription = "重复工单Id", ColumnDataType = "json", IsJson = true, IsNullable = true)]
  139. public List<string>? DuplicateIds { get; set; }
  140. /// <summary>
  141. /// 推送分类
  142. /// </summary>
  143. [SugarColumn(IsNullable = true)]
  144. public string? PushTypeCode { get; set; }
  145. [SugarColumn(IsNullable = true)]
  146. public string? PushType { get; set; }
  147. /// <summary>
  148. /// 附件
  149. /// </summary>
  150. [SugarColumn(ColumnDataType = "json", IsJson = true)]
  151. public List<string> Additions { get; set; } = new();
  152. /// <summary>
  153. /// 诉求内容
  154. /// </summary>
  155. [SugarColumn(ColumnDataType = "varchar(2000)")]
  156. public string Content { get; set; }
  157. #endregion
  158. #region 工单属性
  159. /// <summary>
  160. /// 工单状态
  161. /// </summary>
  162. public EOrderStatus Status { get; set; }
  163. /// <summary>
  164. /// 过期状态 //todo 延迟消息更新此字段
  165. /// </summary>
  166. public EExpiredStatus ExpiredStatus { get; set; }
  167. /// <summary>
  168. /// 来源,区分省平台或110等其他平台同步过来的工单
  169. /// </summary>
  170. public ESource Source { get; set; }
  171. /// <summary>
  172. /// 处理方式(直办、交办)
  173. /// </summary>
  174. [SugarColumn(DefaultValue = "10")]
  175. public EProcessType ProcessType { get; set; } = EProcessType.Zhiban;
  176. ///// <summary>
  177. ///// 进展情况
  178. ///// </summary>
  179. //[SugarColumn(DefaultValue = "0")]
  180. //public EProgress Progress { get; set; }
  181. /// <summary>
  182. /// 是否公开
  183. /// </summary>
  184. [SugarColumn(DefaultValue = "f")]
  185. public bool IsPublicity { get; set; }
  186. /// <summary>
  187. /// 是否为省工单(省派发工单字段:get_case_result_receive == "中国政府网来源工单" 为true)
  188. /// </summary>
  189. [SugarColumn(DefaultValue = "f")]
  190. public bool IsProvince { get; set; }
  191. /// <summary>
  192. /// 工单编码(20220101000001)
  193. /// </summary>
  194. public string No { get; set; }
  195. /// <summary>
  196. /// 省工单编号(冗余)
  197. /// </summary>
  198. [SugarColumn(IsNullable = true)]
  199. public string? FromProvinceNo { get; set; }
  200. /// <summary>
  201. /// 同步省工单编号(按省工单要求生成的编号,冗余)
  202. /// </summary>
  203. [SugarColumn(IsNullable = true)]
  204. public string? ToProvinceNo { get; set; }
  205. /// <summary>
  206. /// 省过期时间(省工单才有)
  207. /// </summary>
  208. public DateTime? ExpiredTimeProvince { get; set; }
  209. /// <summary>
  210. /// 110工单编号(从110来的和推送给110的都用该字段记录唯一标识)
  211. /// </summary>
  212. [SugarColumn(IsNullable = true)]
  213. public string? No110 { get; set; }
  214. /// <summary>
  215. /// 是否已撤销
  216. /// </summary>
  217. [SugarColumn(DefaultValue = "f")]
  218. public bool IsCancel { get; set; }
  219. #endregion
  220. #region 流程信息
  221. /// <summary>
  222. /// 工单开始时间(受理/接办时间=流程开启时间)
  223. /// </summary>
  224. public DateTime? StartTime { get; set; }
  225. /// <summary>
  226. /// 过期时间
  227. /// </summary>
  228. public DateTime? ExpiredTime { get; set; }
  229. /// <summary>
  230. /// 交办时间(中心交部门办理时间)
  231. /// </summary>
  232. public DateTime? CenterToOrgTime { get; set; }
  233. /// <summary>
  234. /// 归档时间(暂为流程结束时间,因流程结束自动归档)
  235. /// </summary>
  236. public DateTime? FiledTime { get; set; }
  237. /// <summary>
  238. /// 办结时长(分钟)
  239. /// 办结时间-交办时间
  240. /// </summary>
  241. [SugarColumn(DefaultValue = "0")]
  242. public double HandleDuration { get; set; }
  243. /// <summary>
  244. /// 办结工作日时长
  245. /// </summary>
  246. [SugarColumn(DefaultValue = "0")]
  247. public double HandleDurationWorkday { get; set; }
  248. /// <summary>
  249. /// 全流程时长(分钟)
  250. /// 归档时间-创建时间
  251. /// </summary>
  252. [SugarColumn(DefaultValue = "0")]
  253. public double AllDuration { get; set; }
  254. /// <summary>
  255. /// 办理时间限制(如:24小时、7个工作日)
  256. /// </summary>
  257. [SugarColumn(IsNullable = true, DefaultValue = "")]
  258. public string? TimeLimit { get; set; }
  259. public int? TimeLimitCount { get; set; }
  260. public ETimeType? TimeLimitUnit { get; set; }
  261. #region 实际办理信息(节点,部门,意见)
  262. /// <summary>
  263. /// 实际办理节点code(会签状态此字段保存最外层会签发起节点code)
  264. /// </summary>
  265. [SugarColumn(IsNullable = true)]
  266. public string? ActualHandleStepCode { get; set; }
  267. /// <summary>
  268. /// 实际办理节点名称(会签状态此字段保存最外层会签发起节点名称)
  269. /// </summary>
  270. [SugarColumn(IsNullable = true)]
  271. public string? ActualHandleStepName { get; set; }
  272. /// <summary>
  273. /// 到达实际办理节点时间(stepBox创建时间)
  274. /// </summary>
  275. public DateTime? ActualHandleStepCreateTime { get; set; }
  276. /// <summary>
  277. /// 实际办理节点签收时间
  278. /// </summary>
  279. public DateTime? ActualHandleStepAcceptTime { get; set; }
  280. /// <summary>
  281. /// 实际办理时间
  282. /// </summary>
  283. public DateTime? ActualHandleTime { get; set; }
  284. /// <summary>
  285. /// 实际办理人id
  286. /// </summary>
  287. [SugarColumn(IsNullable = true)]
  288. public string? ActualHandlerId { get; set; }
  289. /// <summary>
  290. /// 实际办理人名称
  291. /// </summary>
  292. [SugarColumn(IsNullable = true)]
  293. public string? ActualHandlerName { get; set; }
  294. /// <summary>
  295. /// 实际办理部门名称
  296. /// </summary>
  297. [SugarColumn(IsNullable = true)]
  298. public string? ActualHandleOrgName { get; set; }
  299. /// <summary>
  300. /// 实际办理部门编码
  301. /// </summary>
  302. [SugarColumn(IsNullable = true)]
  303. public string? ActualHandleOrgCode { get; set; }
  304. /// <summary>
  305. /// 实际办理部门行政区划编码
  306. /// </summary>
  307. [SugarColumn(IsNullable = true)]
  308. public string? ActualHandleOrgAreaCode { get; set; }
  309. /// <summary>
  310. /// 实际办理部门行政区划名称
  311. /// </summary>
  312. [SugarColumn(IsNullable = true)]
  313. public string? ActualHandleOrgAreaName { get; set; }
  314. /// <summary>
  315. /// 实际办理意见(办理中...or 最终办理意见)
  316. /// </summary>
  317. [SugarColumn(Length = 2000)]
  318. public string ActualOpinion { get; set; } = "办理中...";
  319. /// <summary>
  320. /// 真实办理人姓名(手动填写)
  321. /// </summary>
  322. [SugarColumn(IsNullable = true)]
  323. public string? RealHandlerName { get; set; }
  324. /// <summary>
  325. /// 真实办理人电话(手动填写)
  326. /// </summary>
  327. [SugarColumn(IsNullable = true)]
  328. public string? RealHandlerPhone { get; set; }
  329. /// <summary>
  330. /// 沟通方式(手动填写)
  331. /// </summary>
  332. public ERealCommunicationMode? RealCommunicationMode { get; set; }
  333. /// <summary>
  334. /// 沟通时间(手动填写)
  335. /// </summary>
  336. public DateTime? RealCommunicationTime { get; set; }
  337. /// <summary>
  338. /// 沟通地点(手动填写)
  339. /// </summary>
  340. [SugarColumn(IsNullable = true)]
  341. public string? RealCommunicationAddress { get; set; }
  342. #endregion
  343. #region 当前办理节点信息
  344. /// <summary>
  345. /// 当前办理节点code(非会签:当前被指派节点,会签:会签发起节点)
  346. /// </summary>
  347. [SugarColumn(IsNullable = true)]
  348. public string? CurrentStepCode { get; set; }
  349. /// <summary>
  350. /// 当前节点名称
  351. /// </summary>
  352. [SugarColumn(IsNullable = true)]
  353. public string? CurrentStepName { get; set; }
  354. /// <summary>
  355. /// 到达当前节点时间(stepBox创建时间)
  356. /// </summary>
  357. public DateTime? CurrentStepCreateTime { get; set; }
  358. #endregion
  359. #region 一级部门
  360. /// <summary>
  361. /// 一级部门code
  362. /// </summary>
  363. [SugarColumn(IsNullable = true)]
  364. public string? OrgLevelOneCode { get; set; }
  365. /// <summary>
  366. /// 一级部门名称
  367. /// </summary>
  368. [SugarColumn(IsNullable = true)]
  369. public string? OrgLevelOneName { get; set; }
  370. #endregion
  371. #region 受理人(开启流程的话务员)
  372. /// <summary>
  373. /// 受理人id
  374. /// </summary>
  375. [SugarColumn(IsNullable = true)]
  376. public string? AcceptorId { get; set; }
  377. /// <summary>
  378. /// 受理人名称
  379. /// </summary>
  380. [SugarColumn(IsNullable = true)]
  381. public string? AcceptorName { get; set; }
  382. /// <summary>
  383. /// 受理人工号
  384. /// </summary>
  385. [SugarColumn(IsNullable = true)]
  386. public string? AcceptorStaffNo { get; set; }
  387. /// <summary>
  388. /// 受理人部门编码
  389. /// </summary>
  390. public string? AcceptorOrgCode { get; set; }
  391. /// <summary>
  392. /// 受理人部门名称
  393. /// </summary>
  394. public string? AcceptorOrgName { get; set; }
  395. #endregion
  396. #endregion
  397. /// <summary>
  398. /// 第一次评价结果
  399. /// </summary>
  400. [SugarColumn(IsNullable = true, DefaultValue = "")]
  401. public string? FirstVisitResult { get; set; }
  402. [SugarColumn(IsNullable = true, DefaultValue = "")]
  403. public string? FirstVisitResultCode { get; set; }
  404. }
  405. public partial class Order
  406. {
  407. /// <summary>
  408. /// 受理人
  409. /// </summary>
  410. [Navigate(NavigateType.OneToOne, nameof(AcceptorId))]
  411. public User Acceptor { get; set; }
  412. /// <summary>
  413. /// 热点
  414. /// </summary>
  415. [Navigate(NavigateType.OneToOne, nameof(HotspotId))]
  416. public Hotspot Hotspot { get; set; }
  417. /// <summary>
  418. /// 办理流程
  419. /// </summary>
  420. [Navigate(NavigateType.OneToOne, nameof(WorkflowId))]
  421. public Workflow Workflow { get; set; }
  422. /// <summary>
  423. /// 工单扩展信息(12315-投诉)
  424. /// </summary>
  425. [Navigate(NavigateType.OneToOne, nameof(Id))]
  426. public OrderComplain OrderComplain { get; set; }
  427. /// <summary>
  428. /// 工单扩展信息(12315-举报)
  429. /// </summary>
  430. [Navigate(NavigateType.OneToOne, nameof(Id))]
  431. public OrderReport OrderReport { get; set; }
  432. /// <summary>
  433. /// 已发布工单
  434. /// </summary>
  435. [Navigate(NavigateType.OneToOne, nameof(Id))]
  436. public OrderPublish OrderPublish { get; set; }
  437. /// <summary>
  438. /// 已回访工单
  439. /// </summary>
  440. [Navigate(NavigateType.OneToMany, nameof(OrderVisit.OrderId))]
  441. public List<OrderVisit> OrderVisits { get; set; }
  442. [Navigate(NavigateType.OneToMany, nameof(OrderScreen.OrderId))]
  443. public List<OrderScreen> OrderScreens { get; set; }
  444. #region Method
  445. public void Cancel()
  446. {
  447. IsCancel = true;
  448. }
  449. public void Redo()
  450. {
  451. Status = EOrderStatus.Handling;
  452. ExpiredStatus = EExpiredStatus.Normal;
  453. }
  454. public void CheckIfFiled()
  455. {
  456. if (Status is EOrderStatus.Filed)
  457. throw UserFriendlyException.SameMessage("工单已归档");
  458. }
  459. public void Init(string userId, string? userName, string? staffNo)
  460. {
  461. AcceptorId = userId;
  462. AcceptorName = userName;
  463. AcceptorStaffNo = staffNo;
  464. if (!string.IsNullOrEmpty(Contact))
  465. ContactMask = Contact.MaskPhoneNumber();
  466. Status = EOrderStatus.WaitForAccept;
  467. ExpiredStatus = EExpiredStatus.Normal;
  468. }
  469. /// <summary>
  470. /// 开始工单办理流程
  471. /// </summary>
  472. public void StartManageFlow() => Status = EOrderStatus.Handling;
  473. /// <summary>
  474. /// 更新工单办理中状态
  475. /// </summary>
  476. public void UpdateHandlingStatus(bool isInCountersign) =>
  477. Status = isInCountersign ? EOrderStatus.Countersigning : EOrderStatus.Handling;
  478. ///// <summary>
  479. ///// 接办
  480. ///// </summary>
  481. //public void Accept()
  482. //{
  483. // if (Status is EOrderStatus.Countersigning) return;
  484. // Status = EOrderStatus.Handling;
  485. //}
  486. ///// <summary>
  487. ///// 最终办理
  488. ///// </summary>
  489. //public void FinalManage()
  490. //{
  491. // Status = EOrderStatus.Completed;
  492. //}
  493. ///// <summary>
  494. ///// 撤回最终办理
  495. ///// </summary>
  496. //public void RecallFinalManage()
  497. //{
  498. // Status = EOrderStatus.WaitForSign;
  499. //}
  500. /// <summary>
  501. /// 归档
  502. /// </summary>
  503. public void Filed()
  504. {
  505. if (Status is EOrderStatus.Filed) return;
  506. Status = EOrderStatus.Filed;
  507. FiledTime = DateTime.Now;
  508. }
  509. /// <summary>
  510. /// 发布
  511. /// </summary>
  512. /// <param name="isPublicity"></param>
  513. public void Publish(bool isPublicity)
  514. {
  515. //Progress = EProgress.Published;
  516. Status = EOrderStatus.Published;
  517. IsPublicity = isPublicity;
  518. }
  519. /// <summary>
  520. /// 已回访
  521. /// </summary>
  522. public void Visited(string resultCode, string result)
  523. {
  524. if (string.IsNullOrEmpty(resultCode) || string.IsNullOrEmpty(result))
  525. throw new UserFriendlyException("无效参数");
  526. if (!string.IsNullOrEmpty(FirstVisitResultCode)) return;
  527. FirstVisitResultCode = resultCode;
  528. FirstVisitResult = result;
  529. //Progress = EProgress.Visited;
  530. Status = EOrderStatus.Visited;
  531. }
  532. #endregion
  533. }
  534. }