cnhis-design-vue 2.1.78 → 2.1.80

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (598) hide show
  1. package/.prettierrc.js +27 -27
  2. package/CHANGELOG.md +2953 -2911
  3. package/README.md +90 -90
  4. package/commitlint.config.js +1 -1
  5. package/es/affix/index.js +8 -8
  6. package/es/age/index.js +18 -18
  7. package/es/alert/index.js +8 -8
  8. package/es/anchor/index.js +8 -8
  9. package/es/auto-complete/index.js +8 -8
  10. package/es/avatar/index.js +8 -8
  11. package/es/back-top/index.js +8 -8
  12. package/es/badge/index.js +8 -8
  13. package/es/base/index.js +8 -8
  14. package/es/big-table/index.js +358 -338
  15. package/es/big-table/style.css +1 -1
  16. package/es/breadcrumb/index.js +8 -8
  17. package/es/button/index.js +31 -31
  18. package/es/calendar/index.js +8 -8
  19. package/es/captcha/index.js +7 -7
  20. package/es/card/index.js +8 -8
  21. package/es/carousel/index.js +8 -8
  22. package/es/cascader/index.js +8 -8
  23. package/es/checkbox/index.js +17 -17
  24. package/es/col/index.js +8 -8
  25. package/es/collapse/index.js +8 -8
  26. package/es/color-picker/index.js +5 -5
  27. package/es/comment/index.js +8 -8
  28. package/es/config-provider/index.js +8 -8
  29. package/es/date-picker/index.js +30 -30
  30. package/es/descriptions/index.js +8 -8
  31. package/es/divider/index.js +8 -8
  32. package/es/drag-layout/index.js +5 -5
  33. package/es/drawer/index.js +8 -8
  34. package/es/dropdown/index.js +8 -8
  35. package/es/editor/index.js +1 -1
  36. package/es/ellipsis/index.js +1 -1
  37. package/es/empty/index.js +8 -8
  38. package/es/fabric-chart/index.js +45 -45
  39. package/es/form/index.js +8 -8
  40. package/es/form-model/index.js +8 -8
  41. package/es/form-table/index.js +144 -144
  42. package/es/grid/index.js +4 -4
  43. package/es/index/index.js +3657 -2281
  44. package/es/index/style.css +1 -1
  45. package/es/input/index.js +11 -11
  46. package/es/input-number/index.js +8 -8
  47. package/es/keep-cache/index.js +9 -9
  48. package/es/layout/index.js +8 -8
  49. package/es/list/index.js +8 -8
  50. package/es/locale-provider/index.js +8 -8
  51. package/es/map/index.js +13 -13
  52. package/es/mentions/index.js +8 -8
  53. package/es/menu/index.js +8 -8
  54. package/es/message/index.js +8 -8
  55. package/es/multi-chat/index.js +117 -117
  56. package/es/multi-chat-client/index.js +111 -111
  57. package/es/multi-chat-history/index.js +6 -6
  58. package/es/multi-chat-record/index.js +27 -27
  59. package/es/multi-chat-setting/index.js +43 -43
  60. package/es/multi-chat-sip/index.js +1 -1
  61. package/es/notification/index.js +8 -8
  62. package/es/page-header/index.js +8 -8
  63. package/es/pagination/index.js +8 -8
  64. package/es/popconfirm/index.js +8 -8
  65. package/es/popover/index.js +8 -8
  66. package/es/progress/index.js +8 -8
  67. package/es/radio/index.js +17 -17
  68. package/es/rate/index.js +8 -8
  69. package/es/result/index.js +8 -8
  70. package/es/row/index.js +8 -8
  71. package/es/scale-container/index.js +1 -1
  72. package/es/scale-view/index.js +200 -198
  73. package/es/scale-view/style.css +1 -1
  74. package/es/select/index.js +44 -44
  75. package/es/select-label/index.js +75 -73
  76. package/es/select-label/style.css +1 -1
  77. package/es/select-person/index.js +45 -41
  78. package/es/select-person/style.css +1 -1
  79. package/es/shortcut-setter/index.js +10 -10
  80. package/es/skeleton/index.js +8 -8
  81. package/es/slider/index.js +8 -8
  82. package/es/space/index.js +8 -8
  83. package/es/spin/index.js +8 -8
  84. package/es/statistic/index.js +8 -8
  85. package/es/steps/index.js +8 -8
  86. package/es/switch/index.js +8 -8
  87. package/es/table-filter/index.js +2697 -1347
  88. package/es/table-filter/style.css +1 -1
  89. package/es/tabs/index.js +8 -8
  90. package/es/tag/index.js +9 -9
  91. package/es/time-picker/index.js +8 -8
  92. package/es/timeline/index.js +8 -8
  93. package/es/tooltip/index.js +8 -8
  94. package/es/transfer/index.js +8 -8
  95. package/es/tree/index.js +8 -8
  96. package/es/tree-select/index.js +8 -8
  97. package/es/upload/index.js +8 -8
  98. package/es/utils/clickoutside.js +7 -7
  99. package/es/utils/kty.min-1.0.0.js +5 -5
  100. package/es/utils/time-domain.js +16 -16
  101. package/es/utils/utils-map.js +32 -32
  102. package/es/utils/vexutils.js +27 -27
  103. package/es/verification-code/index.js +2 -2
  104. package/lib/cui.common.js +11438 -9918
  105. package/lib/cui.umd.js +11438 -9918
  106. package/lib/cui.umd.min.js +69 -69
  107. package/package.json +110 -110
  108. package/packages/affix/index.js +12 -12
  109. package/packages/affix/src/Affix.jsx +12 -12
  110. package/packages/age/index.js +27 -27
  111. package/packages/age/src/age/index.vue +139 -139
  112. package/packages/age/src/age-com/index.vue +209 -209
  113. package/packages/alert/index.js +12 -12
  114. package/packages/alert/src/Alert.jsx +11 -11
  115. package/packages/anchor/index.js +18 -18
  116. package/packages/anchor/src/Anchor.jsx +11 -11
  117. package/packages/anchor/src/Link.jsx +11 -11
  118. package/packages/auto-complete/index.js +12 -12
  119. package/packages/auto-complete/src/AutoComplete.jsx +12 -12
  120. package/packages/avatar/index.js +12 -12
  121. package/packages/avatar/src/Avatar.jsx +11 -11
  122. package/packages/back-top/index.js +12 -12
  123. package/packages/back-top/src/BackTop.jsx +11 -11
  124. package/packages/badge/index.js +12 -12
  125. package/packages/badge/src/Badge.jsx +11 -11
  126. package/packages/base/index.js +3 -3
  127. package/packages/big-table/index.js +16 -16
  128. package/packages/big-table/src/BigTable.vue +3205 -3193
  129. package/packages/big-table/src/Fieldset.vue +2067 -2067
  130. package/packages/big-table/src/assets/iconfont/iconfont.css +21 -21
  131. package/packages/big-table/src/assets/style/table-base.less +390 -387
  132. package/packages/big-table/src/assets/style/table-global.less +175 -175
  133. package/packages/big-table/src/components/AutoLayoutButton.vue +307 -307
  134. package/packages/big-table/src/components/NoData.vue +81 -81
  135. package/packages/big-table/src/components/TextOverTooltip.vue +120 -120
  136. package/packages/big-table/src/components/edit-form/EditForm.vue +509 -509
  137. package/packages/big-table/src/components/edit-form/edit-component/config-data/index.js +68 -68
  138. package/packages/big-table/src/components/edit-form/edit-component/edit-date-picker/edit-date-picker.vue +60 -60
  139. package/packages/big-table/src/components/edit-form/edit-component/edit-digital/edit-digital.vue +54 -54
  140. package/packages/big-table/src/components/edit-form/edit-component/edit-input/edit-input.vue +39 -39
  141. package/packages/big-table/src/components/edit-form/edit-component/edit-input-password/edit-input-password.vue +79 -79
  142. package/packages/big-table/src/components/edit-form/edit-component/edit-month-picker/edit-month-picker.vue +37 -37
  143. package/packages/big-table/src/components/edit-form/edit-component/edit-search/edit-search.vue +82 -82
  144. package/packages/big-table/src/components/edit-form/edit-component/edit-search-more/edit-search-more.vue +95 -95
  145. package/packages/big-table/src/components/edit-form/edit-component/edit-select/edit-select.vue +40 -40
  146. package/packages/big-table/src/components/edit-form/edit-component/edit-select-multiple/edit-select-multiple.vue +55 -55
  147. package/packages/big-table/src/components/edit-form/edit-component/edit-switch/edit-switch.vue +43 -43
  148. package/packages/big-table/src/components/edit-form/edit-component/edit-textarea/edit-textarea.vue +41 -41
  149. package/packages/big-table/src/components/edit-form/edit-component/edit-time-picker/edit-time-picker.vue +40 -40
  150. package/packages/big-table/src/components/edit-form/edit-component/mixins/bound-date.js +3 -3
  151. package/packages/big-table/src/components/edit-form/edit-component/mixins/dateType.js +217 -217
  152. package/packages/big-table/src/components/edit-form/edit-component/mixins/item-default.js +639 -639
  153. package/packages/big-table/src/components/edit-form/edit-component/mixins/search.js +1247 -1247
  154. package/packages/big-table/src/components/edit-form/edit-component/register-com.js +26 -26
  155. package/packages/big-table/src/components/edit-form/edit-item/form-event.js +80 -80
  156. package/packages/big-table/src/components/edit-form/edit-item/global-props.js +33 -33
  157. package/packages/big-table/src/components/edit-form/edit-item/index.js +4 -4
  158. package/packages/big-table/src/components/edit-form/edit-item/render-methods.js +28 -28
  159. package/packages/big-table/src/components/edit-form/edit-item/validate-rules.js +463 -463
  160. package/packages/big-table/src/components/edit-form/edit-mixins/form-commom.js +673 -673
  161. package/packages/big-table/src/components/edit-form/edit-mixins/index.js +3 -3
  162. package/packages/big-table/src/components/edit-form/edit-utils/index.js +112 -112
  163. package/packages/big-table/src/components/password-com.vue +58 -58
  164. package/packages/big-table/src/components/player-vod/index.vue +57 -57
  165. package/packages/big-table/src/components/player-vod/player.vue +193 -193
  166. package/packages/big-table/src/components/player-vod/video-list.vue +265 -265
  167. package/packages/big-table/src/components/player-vod/video-modal.vue +126 -126
  168. package/packages/big-table/src/utils/CustomPagination.vue +86 -86
  169. package/packages/big-table/src/utils/batchEditing.js +610 -610
  170. package/packages/big-table/src/utils/bigTableProps.js +108 -108
  171. package/packages/big-table/src/utils/eventBroadcast.js +24 -24
  172. package/packages/big-table/src/utils/format.js +557 -557
  173. package/packages/big-table/src/utils/nestTable.js +109 -109
  174. package/packages/big-table/src/utils/tableParse.js +234 -234
  175. package/packages/breadcrumb/index.js +21 -21
  176. package/packages/breadcrumb/src/Breadcrumb.jsx +11 -11
  177. package/packages/breadcrumb/src/BreadcrumbItem.jsx +11 -11
  178. package/packages/breadcrumb/src/BreadcrumbSeparator.jsx +11 -11
  179. package/packages/button/index.js +21 -21
  180. package/packages/button/src/Button.jsx +11 -11
  181. package/packages/button/src/ButtonGroup.jsx +11 -11
  182. package/packages/button/src/ButtonPrint/components/IdentityVerification.vue +181 -181
  183. package/packages/button/src/ButtonPrint/index.vue +766 -766
  184. package/packages/button/src/ButtonPrint/mixin/his-print.js +95 -95
  185. package/packages/calendar/index.js +12 -12
  186. package/packages/calendar/src/Calendar.jsx +11 -11
  187. package/packages/captcha/index.js +9 -9
  188. package/packages/captcha/src/Captcha.vue +164 -164
  189. package/packages/captcha/src/Index.vue +47 -47
  190. package/packages/captcha/src/SlideVerify.vue +285 -285
  191. package/packages/card/index.js +21 -21
  192. package/packages/card/src/Card.jsx +12 -12
  193. package/packages/card/src/CardGrid.js +7 -7
  194. package/packages/card/src/CardMeta.js +7 -7
  195. package/packages/carousel/index.js +12 -12
  196. package/packages/carousel/src/Carousel.jsx +12 -12
  197. package/packages/cascader/index.js +12 -12
  198. package/packages/cascader/src/Cascader.jsx +11 -11
  199. package/packages/checkbox/index.js +30 -30
  200. package/packages/checkbox/src/Checkbox.jsx +11 -11
  201. package/packages/checkbox/src/CheckboxImg/index.vue +141 -141
  202. package/packages/checkbox/src/Group.jsx +11 -11
  203. package/packages/col/index.js +13 -13
  204. package/packages/col/src/Col.jsx +11 -11
  205. package/packages/collapse/index.js +18 -18
  206. package/packages/collapse/src/Collapse.jsx +11 -11
  207. package/packages/collapse/src/Panel.jsx +11 -11
  208. package/packages/color-picker/index.js +10 -10
  209. package/packages/color-picker/src/color-picker.vue +191 -191
  210. package/packages/color-picker/src/style.less +109 -109
  211. package/packages/comment/index.js +12 -12
  212. package/packages/comment/src/Comment.jsx +11 -11
  213. package/packages/config-provider/index.js +12 -12
  214. package/packages/config-provider/src/ConfigProvider.jsx +11 -11
  215. package/packages/date-picker/index.js +26 -26
  216. package/packages/date-picker/src/DatePicker.jsx +12 -12
  217. package/packages/date-picker/src/MonthPicker.jsx +11 -11
  218. package/packages/date-picker/src/RangePicker.jsx +11 -11
  219. package/packages/date-picker/src/WeekPicker.jsx +11 -11
  220. package/packages/date-picker/src/utils/index.js +374 -374
  221. package/packages/descriptions/index.js +18 -18
  222. package/packages/descriptions/src/Descriptions.jsx +11 -11
  223. package/packages/descriptions/src/Item.jsx +11 -11
  224. package/packages/divider/index.js +12 -12
  225. package/packages/divider/src/Divider.jsx +11 -11
  226. package/packages/drag-layout/DragFormLeftItem.vue +173 -173
  227. package/packages/drag-layout/DragFormRightItem.vue +284 -284
  228. package/packages/drag-layout/I18n-mixins.js +10 -10
  229. package/packages/drag-layout/drag-layout.vue +778 -778
  230. package/packages/drag-layout/index.js +12 -12
  231. package/packages/drawer/index.js +12 -12
  232. package/packages/drawer/src/Drawer.jsx +11 -11
  233. package/packages/dropdown/index.js +12 -12
  234. package/packages/dropdown/src/Dropdown.jsx +11 -11
  235. package/packages/editor/index.js +9 -9
  236. package/packages/ellipsis/index.js +8 -8
  237. package/packages/ellipsis/src/Ellipsis.vue +65 -65
  238. package/packages/empty/index.js +12 -12
  239. package/packages/empty/src/Empty.jsx +11 -11
  240. package/packages/fabric-chart/index.js +9 -9
  241. package/packages/fabric-chart/src/FabricGrid.vue +67 -67
  242. package/packages/fabric-chart/src/components/DropPopup.vue +90 -90
  243. package/packages/fabric-chart/src/components/MouseRightClick.vue +168 -168
  244. package/packages/fabric-chart/src/components/TimeScaleValue.vue +125 -125
  245. package/packages/fabric-chart/src/const/defaultVaule.js +60 -60
  246. package/packages/fabric-chart/src/fabric-chart/FabricLines.vue +617 -617
  247. package/packages/fabric-chart/src/fabric-chart/FabricPolylines.vue +1191 -1191
  248. package/packages/fabric-chart/src/fabric-chart/FabricScaleValue.vue +139 -139
  249. package/packages/fabric-chart/src/fabric-chart/FabricTextGroup.vue +651 -651
  250. package/packages/fabric-chart/src/mixins/fabricCommon.js +81 -81
  251. package/packages/fabric-chart/src/mixins/fabricObject.js +193 -193
  252. package/packages/fabric-chart/src/mixins/type.js +5 -5
  253. package/packages/fabric-chart/src/utils/bus.js +2 -2
  254. package/packages/form/index.js +16 -16
  255. package/packages/form/src/Form.jsx +11 -11
  256. package/packages/form/src/Item.jsx +11 -11
  257. package/packages/form-model/index.js +14 -14
  258. package/packages/form-model/src/FormModel.jsx +11 -11
  259. package/packages/form-model/src/Item.jsx +11 -11
  260. package/packages/form-table/index.js +16 -16
  261. package/packages/form-table/src/FormTable.vue +1110 -1110
  262. package/packages/form-table/src/components/table-component/config-data/index.js +80 -80
  263. package/packages/form-table/src/components/table-component/global-props.js +22 -22
  264. package/packages/form-table/src/components/table-component/index.js +8 -8
  265. package/packages/form-table/src/components/table-component/mixins/bound-date.js +455 -455
  266. package/packages/form-table/src/components/table-component/mixins/dateType.js +217 -217
  267. package/packages/form-table/src/components/table-component/mixins/item-default.js +257 -257
  268. package/packages/form-table/src/components/table-component/mixins/search.js +1242 -1242
  269. package/packages/form-table/src/components/table-component/register-com.js +30 -30
  270. package/packages/form-table/src/components/table-component/table-age/table-age.vue +175 -175
  271. package/packages/form-table/src/components/table-component/table-date-picker/table-date-picker.vue +87 -87
  272. package/packages/form-table/src/components/table-component/table-digital/table-digital.vue +93 -93
  273. package/packages/form-table/src/components/table-component/table-input/table-input.vue +81 -81
  274. package/packages/form-table/src/components/table-component/table-input-password/table-input-password.vue +126 -126
  275. package/packages/form-table/src/components/table-component/table-month-picker/table-month-picker.vue +55 -55
  276. package/packages/form-table/src/components/table-component/table-search/table-search.vue +174 -174
  277. package/packages/form-table/src/components/table-component/table-search-more/table-search-more.vue +191 -191
  278. package/packages/form-table/src/components/table-component/table-select/table-select.vue +64 -64
  279. package/packages/form-table/src/components/table-component/table-select-multiple/table-select-multiple.vue +81 -81
  280. package/packages/form-table/src/components/table-component/table-textarea/table-textarea.vue +76 -76
  281. package/packages/form-table/src/components/table-component/table-time-picker/table-time-picker.vue +55 -55
  282. package/packages/form-table/src/components/table-component/table-tree-select/table-tree-select.vue +135 -135
  283. package/packages/form-table/src/components/table-component/text-over-tooltip/TextOverTooltip.vue +97 -97
  284. package/packages/form-table/src/components/table-item/form-event.js +81 -81
  285. package/packages/form-table/src/components/table-item/global-props.js +27 -27
  286. package/packages/form-table/src/components/table-item/index.js +4 -4
  287. package/packages/form-table/src/components/table-item/render-methods.js +28 -28
  288. package/packages/form-table/src/components/table-item/validate-rules.js +520 -520
  289. package/packages/form-table/src/components/table-mixins/form-commom.js +98 -98
  290. package/packages/form-table/src/components/table-mixins/index.js +3 -3
  291. package/packages/form-table/src/components/table-utils/index.js +112 -112
  292. package/packages/form-table/src/disabledDetail.less +46 -46
  293. package/packages/grid/index.js +10 -10
  294. package/packages/grid/src/grid.js +28 -28
  295. package/packages/icon/index.js +10 -10
  296. package/packages/icon/src/icon.js +13 -13
  297. package/packages/index.js +313 -313
  298. package/packages/input/index.js +30 -30
  299. package/packages/input/src/Group.jsx +11 -11
  300. package/packages/input/src/Input.jsx +11 -11
  301. package/packages/input/src/Password.jsx +11 -11
  302. package/packages/input/src/Search.jsx +11 -11
  303. package/packages/input/src/TextArea.jsx +11 -11
  304. package/packages/input/src/input-quick/components/quick-item.vue +284 -284
  305. package/packages/input/src/input-quick/components/quick-popover.vue +596 -596
  306. package/packages/input/src/input-quick/index.vue +137 -137
  307. package/packages/input-number/index.js +12 -12
  308. package/packages/input-number/src/InputNumber.jsx +11 -11
  309. package/packages/keep-cache/KeepCache.js +236 -236
  310. package/packages/keep-cache/index.css +2 -2
  311. package/packages/keep-cache/index.js +8 -8
  312. package/packages/layout/index.js +27 -27
  313. package/packages/layout/src/Content.jsx +11 -11
  314. package/packages/layout/src/Footer.jsx +11 -11
  315. package/packages/layout/src/Header.jsx +11 -11
  316. package/packages/layout/src/Layout.jsx +11 -11
  317. package/packages/layout/src/Sider.jsx +11 -11
  318. package/packages/list/index.js +21 -21
  319. package/packages/list/src/Item.jsx +11 -11
  320. package/packages/list/src/ItemMeta.jsx +11 -11
  321. package/packages/list/src/List.jsx +11 -11
  322. package/packages/locale-provider/index.js +12 -12
  323. package/packages/locale-provider/src/LocaleProvider.jsx +11 -11
  324. package/packages/map/index.js +9 -9
  325. package/packages/map/src/Map.vue +484 -484
  326. package/packages/map/src/popup-map.vue +53 -53
  327. package/packages/mentions/index.js +18 -18
  328. package/packages/mentions/src/Mentions.jsx +11 -11
  329. package/packages/mentions/src/Option.jsx +11 -11
  330. package/packages/menu/index.js +27 -27
  331. package/packages/menu/src/Divider.jsx +11 -11
  332. package/packages/menu/src/Item.jsx +11 -11
  333. package/packages/menu/src/ItemGroup.jsx +11 -11
  334. package/packages/menu/src/Menu.jsx +11 -11
  335. package/packages/menu/src/SubMenu.jsx +11 -11
  336. package/packages/message/index.js +8 -8
  337. package/packages/modal/index.js +10 -10
  338. package/packages/modal/src/Modal.js +7 -7
  339. package/packages/multi-chat/chat/addConference.vue +200 -200
  340. package/packages/multi-chat/chat/addMembers.vue +411 -411
  341. package/packages/multi-chat/chat/advancedFilter.vue +372 -372
  342. package/packages/multi-chat/chat/calling.vue +246 -246
  343. package/packages/multi-chat/chat/chatFooter.vue +1647 -1647
  344. package/packages/multi-chat/chat/chatHistory.vue +605 -605
  345. package/packages/multi-chat/chat/chatMain.vue +1490 -1490
  346. package/packages/multi-chat/chat/client/index.vue +149 -149
  347. package/packages/multi-chat/chat/delay.vue +177 -177
  348. package/packages/multi-chat/chat/evaluate.vue +343 -343
  349. package/packages/multi-chat/chat/messageRecord.vue +324 -324
  350. package/packages/multi-chat/chat/mixins/NoData.js +20 -20
  351. package/packages/multi-chat/chat/mixins/base.js +97 -97
  352. package/packages/multi-chat/chat/mixins/uniRTCAPI.js +80 -80
  353. package/packages/multi-chat/chat/mixins/viewerOptions.js +67 -67
  354. package/packages/multi-chat/chat/quickReply.vue +439 -439
  355. package/packages/multi-chat/chat/robot/index.vue +312 -312
  356. package/packages/multi-chat/chat/scrollList.vue +1259 -1259
  357. package/packages/multi-chat/chat/videoVoiceList.vue +348 -348
  358. package/packages/multi-chat/chat/voice.vue +431 -431
  359. package/packages/multi-chat/components/avatar.vue +113 -113
  360. package/packages/multi-chat/components/chat-tabs-header.vue +251 -251
  361. package/packages/multi-chat/components/classify-tabs.vue +185 -185
  362. package/packages/multi-chat/components/empty.vue +24 -24
  363. package/packages/multi-chat/components/modal-refuse-reason.vue +112 -112
  364. package/packages/multi-chat/components/modal-sip.vue +160 -160
  365. package/packages/multi-chat/components/modal-user-transfer.vue +98 -98
  366. package/packages/multi-chat/components/msg-describe.vue +138 -138
  367. package/packages/multi-chat/components/msg-picture.vue +68 -68
  368. package/packages/multi-chat/components/msg-prescription.vue +205 -205
  369. package/packages/multi-chat/components/read-record.vue +133 -133
  370. package/packages/multi-chat/components/read-status.vue +34 -34
  371. package/packages/multi-chat/components/user-status.vue +198 -198
  372. package/packages/multi-chat/components/vuecmf-dialog.vue +322 -322
  373. package/packages/multi-chat/index.js +7 -7
  374. package/packages/multi-chat/setting/authority/index.vue +156 -156
  375. package/packages/multi-chat/setting/authority/roleSetting.vue +204 -204
  376. package/packages/multi-chat/setting/baseInfo/index.vue +1343 -1343
  377. package/packages/multi-chat/setting/customerService/batchSelect.vue +403 -403
  378. package/packages/multi-chat/setting/customerService/index.vue +273 -273
  379. package/packages/multi-chat/setting/event/edit/condition.vue +128 -128
  380. package/packages/multi-chat/setting/event/edit/index.vue +437 -437
  381. package/packages/multi-chat/setting/event/edit/notice.vue +129 -129
  382. package/packages/multi-chat/setting/event/edit/strategy.vue +98 -98
  383. package/packages/multi-chat/setting/event/index.vue +249 -249
  384. package/packages/multi-chat/setting/index.vue +269 -269
  385. package/packages/multi-chat/setting/page.vue +14 -14
  386. package/packages/multi-chat/setting/sessionList/index.vue +412 -412
  387. package/packages/multi-chat/setting/sessionList/messageRecord.vue +372 -372
  388. package/packages/multi-chat/setting/userConfig/index.vue +124 -124
  389. package/packages/multi-chat/setting/worktime/index.vue +274 -274
  390. package/packages/multi-chat/store/actions.js +452 -452
  391. package/packages/multi-chat/store/getters.js +380 -380
  392. package/packages/multi-chat/store/helper.js +66 -66
  393. package/packages/multi-chat/store/index.js +50 -50
  394. package/packages/multi-chat/store/mutation.js +305 -305
  395. package/packages/multi-chat/store/state.js +120 -120
  396. package/packages/multi-chat/style/emoji.css +315 -315
  397. package/packages/multi-chat/style/message.mixin.less +38 -38
  398. package/packages/multi-chat/utils/chatSock.js +93 -93
  399. package/packages/multi-chat/utils/compressImage.js +115 -115
  400. package/packages/multi-chat/utils/emoji.json +68 -68
  401. package/packages/multi-chat/utils/index.js +259 -259
  402. package/packages/multi-chat/utils/observer-scroll.js +49 -49
  403. package/packages/multi-chat/utils/panelsetting.js +48 -48
  404. package/packages/multi-chat-client/index.js +7 -7
  405. package/packages/multi-chat-history/index.js +7 -7
  406. package/packages/multi-chat-record/index.js +7 -7
  407. package/packages/multi-chat-setting/index.js +7 -7
  408. package/packages/multi-chat-sip/index.js +6 -6
  409. package/packages/notification/index.js +8 -8
  410. package/packages/page-header/index.js +12 -12
  411. package/packages/page-header/src/PageHeader.jsx +11 -11
  412. package/packages/pagination/index.js +12 -12
  413. package/packages/pagination/src/Pagination.jsx +11 -11
  414. package/packages/popconfirm/index.js +12 -12
  415. package/packages/popconfirm/src/Popconfirm.jsx +11 -11
  416. package/packages/popover/index.js +12 -12
  417. package/packages/popover/src/Popover.jsx +11 -11
  418. package/packages/progress/index.js +12 -12
  419. package/packages/progress/src/Progress.jsx +11 -11
  420. package/packages/radio/index.js +33 -33
  421. package/packages/radio/src/Group.jsx +11 -11
  422. package/packages/radio/src/Radio.jsx +11 -11
  423. package/packages/radio/src/RadioButton.jsx +11 -11
  424. package/packages/radio/src/RadioImg/index.vue +124 -124
  425. package/packages/rate/index.js +12 -12
  426. package/packages/rate/src/Rate.jsx +11 -11
  427. package/packages/result/index.js +12 -12
  428. package/packages/result/src/Result.jsx +11 -11
  429. package/packages/row/index.js +12 -12
  430. package/packages/row/src/Row.jsx +11 -11
  431. package/packages/scale-container/index.js +8 -8
  432. package/packages/scale-container/src/ScaleContainer.vue +197 -197
  433. package/packages/scale-view/NoData.vue +81 -81
  434. package/packages/scale-view/answerParse.vue +133 -133
  435. package/packages/scale-view/customList.vue +801 -801
  436. package/packages/scale-view/data.js +80 -80
  437. package/packages/scale-view/evaluateCountdown.vue +155 -155
  438. package/packages/scale-view/evaluatePage.vue +202 -202
  439. package/packages/scale-view/formitem/data.js +3991 -3991
  440. package/packages/scale-view/formitem/index.js +7 -7
  441. package/packages/scale-view/formitem/r-address.vue +245 -245
  442. package/packages/scale-view/formitem/r-choice.vue +746 -746
  443. package/packages/scale-view/formitem/r-input.vue +93 -93
  444. package/packages/scale-view/formitem/r-prompt.vue +52 -52
  445. package/packages/scale-view/formitem/r-sign.vue +218 -218
  446. package/packages/scale-view/formitem/r-time.vue +285 -285
  447. package/packages/scale-view/formitem/r-upload-custom-list.vue +242 -242
  448. package/packages/scale-view/formitem/r-upload.vue +287 -287
  449. package/packages/scale-view/formitem/sign-com.vue +316 -316
  450. package/packages/scale-view/formitem/text-over-tooltip/TextOverTooltip.vue +98 -98
  451. package/packages/scale-view/index.js +17 -17
  452. package/packages/scale-view/mixin/NoData.js +38 -38
  453. package/packages/scale-view/mixin/addressVal.js +156 -156
  454. package/packages/scale-view/mixin/evaluate.js +146 -146
  455. package/packages/scale-view/mixin/index.js +337 -337
  456. package/packages/scale-view/mixin/judgeTypes.js +267 -267
  457. package/packages/scale-view/scaleView.vue +2017 -2017
  458. package/packages/select/index.js +27 -27
  459. package/packages/select/src/CustomSelect/index.vue +130 -130
  460. package/packages/select/src/OptGroup.jsx +11 -11
  461. package/packages/select/src/Option.jsx +11 -11
  462. package/packages/select/src/Select/Select.vue +231 -231
  463. package/packages/select/src/Select/index.js +12 -12
  464. package/packages/select/src/TableSelect/index.vue +514 -514
  465. package/packages/select-label/index.js +14 -14
  466. package/packages/select-label/label-classify.vue +129 -129
  467. package/packages/select-label/labelFormContent.vue +787 -787
  468. package/packages/select-label/select-label.vue +597 -581
  469. package/packages/select-person/index.js +10 -10
  470. package/packages/select-person/search-tree.vue +373 -373
  471. package/packages/select-person/select-person.vue +1703 -1696
  472. package/packages/shortcut-setter/index.js +12 -12
  473. package/packages/shortcut-setter/src/ShortcutSetter.vue +55 -55
  474. package/packages/shortcut-setter/src/ShortcutSetterItem.vue +85 -85
  475. package/packages/shortcut-setter/src/utils/index.js +63 -63
  476. package/packages/skeleton/index.js +12 -12
  477. package/packages/skeleton/src/Skeleton.jsx +11 -11
  478. package/packages/slider/index.js +12 -12
  479. package/packages/slider/src/Slider.jsx +11 -11
  480. package/packages/space/index.js +12 -12
  481. package/packages/space/src/Space.jsx +11 -11
  482. package/packages/spin/index.js +12 -12
  483. package/packages/spin/src/Spin.jsx +11 -11
  484. package/packages/statistic/index.js +18 -18
  485. package/packages/statistic/src/Countdown.jsx +11 -11
  486. package/packages/statistic/src/Statistic.jsx +11 -11
  487. package/packages/steps/index.js +18 -18
  488. package/packages/steps/src/Step.jsx +11 -11
  489. package/packages/steps/src/Steps.jsx +11 -11
  490. package/packages/switch/index.js +12 -12
  491. package/packages/switch/src/Switch.jsx +11 -11
  492. package/packages/table-filter/index.js +27 -27
  493. package/packages/table-filter/src/base-search-com/BaseSearch.vue +2622 -2630
  494. package/packages/table-filter/src/classification/Classification-com.vue +1760 -1760
  495. package/packages/table-filter/src/classification/search-class-name.vue +266 -266
  496. package/packages/table-filter/src/classification/search-professional-model.vue +680 -680
  497. package/packages/table-filter/src/components/TextOverTooltip.vue +120 -107
  498. package/packages/table-filter/src/components/age-com/index.vue +205 -205
  499. package/packages/table-filter/src/components/button-group/ButtonGroup.vue +162 -162
  500. package/packages/table-filter/src/components/button-icon/button-icon.js +33 -33
  501. package/packages/table-filter/src/components/button-icon/getBtnIcon.js +34 -34
  502. package/packages/table-filter/src/components/c-tree-select/tree-select.vue +336 -336
  503. package/packages/table-filter/src/components/checkbox-group/CheckboxGroup.vue +53 -0
  504. package/packages/table-filter/src/components/drop-button/drop-button.vue +224 -224
  505. package/packages/table-filter/src/components/drop-button/head-btn-icon.js +33 -33
  506. package/packages/table-filter/src/components/drop-view/drop-view.vue +89 -89
  507. package/packages/table-filter/src/components/multi-select/multi-select.vue +242 -236
  508. package/packages/table-filter/src/components/out-quick-search/out-quick-search.vue +232 -346
  509. package/packages/table-filter/src/components/range-age/index.vue +171 -172
  510. package/packages/table-filter/src/components/render-widget/components/CheckboxGroup.vue +51 -51
  511. package/packages/table-filter/src/components/render-widget/components/DateRangeQuick.vue +212 -0
  512. package/packages/table-filter/src/components/render-widget/components/NumberRange.vue +145 -0
  513. package/packages/table-filter/src/components/render-widget/components/Select.vue +50 -50
  514. package/packages/table-filter/src/components/render-widget/components/SelectDynamic.vue +304 -0
  515. package/packages/table-filter/src/components/render-widget/components/index.js +7 -7
  516. package/packages/table-filter/src/components/render-widget/enums.js +34 -29
  517. package/packages/table-filter/src/components/render-widget/index.vue +124 -91
  518. package/packages/table-filter/src/components/render-widget/mixins/dynamic-method.js +158 -0
  519. package/packages/table-filter/src/components/render-widget/widgetCfgMaps.js +245 -140
  520. package/packages/table-filter/src/components/search-condition/SearchCondition.vue +1907 -1907
  521. package/packages/table-filter/src/components/search-condition/fieldTypeList.js +169 -169
  522. package/packages/table-filter/src/components/search-filter/SearchFilter.vue +280 -278
  523. package/packages/table-filter/src/components/search-modal/set-classification.vue +336 -336
  524. package/packages/table-filter/src/components/table-modal/TableModal.vue +477 -477
  525. package/packages/table-filter/src/const/dataOptions.js +43 -43
  526. package/packages/table-filter/src/const/index.js +1 -1
  527. package/packages/table-filter/src/index.vue +584 -584
  528. package/packages/table-filter/src/mixins/mixins.js +773 -714
  529. package/packages/table-filter/src/mixins/out-quick-method.js +123 -0
  530. package/packages/table-filter/src/mixins/renderWidget.js +97 -89
  531. package/packages/table-filter/src/mixins/tableSearchCon.js +128 -128
  532. package/packages/table-filter/src/mixins/wordBookutils.js +113 -113
  533. package/packages/table-filter/src/quick-search/QuickSearch.vue +2190 -2167
  534. package/packages/tabs/index.js +18 -18
  535. package/packages/tabs/src/TabPane.jsx +11 -11
  536. package/packages/tabs/src/Tabs.jsx +11 -11
  537. package/packages/tag/index.js +21 -21
  538. package/packages/tag/src/CheckableTag.jsx +11 -11
  539. package/packages/tag/src/Tag.jsx +11 -11
  540. package/packages/tag/src/TagGroup.vue +621 -621
  541. package/packages/time-picker/index.js +12 -12
  542. package/packages/time-picker/src/TimePicker.jsx +11 -11
  543. package/packages/timeline/index.js +14 -14
  544. package/packages/timeline/src/Item.jsx +11 -11
  545. package/packages/timeline/src/Timeline.jsx +11 -11
  546. package/packages/tooltip/index.js +12 -12
  547. package/packages/tooltip/src/Tooltip.jsx +11 -11
  548. package/packages/transfer/index.js +12 -12
  549. package/packages/transfer/src/Transfer.jsx +11 -11
  550. package/packages/tree/index.js +18 -18
  551. package/packages/tree/src/Tree.jsx +11 -11
  552. package/packages/tree/src/TreeNode.jsx +11 -11
  553. package/packages/tree-select/index.js +18 -18
  554. package/packages/tree-select/src/TreeNode.jsx +11 -11
  555. package/packages/tree-select/src/TreeSelect.jsx +11 -11
  556. package/packages/upload/chunk-upload/chunk-upload-new.vue +1001 -1001
  557. package/packages/upload/chunk-upload/vod-chunk-upload.vue +749 -749
  558. package/packages/upload/chunk-upload/vod-upload-modal.vue +100 -100
  559. package/packages/upload/index.js +12 -12
  560. package/packages/upload/src/Upload.jsx +11 -11
  561. package/packages/verification-code/SlideVerify.vue +306 -306
  562. package/packages/verification-code/index.js +17 -17
  563. package/packages/verification-code/verification-code.vue +147 -147
  564. package/src/component/player-vod/index.vue +57 -57
  565. package/src/component/player-vod/player.vue +192 -192
  566. package/src/component/player-vod/video-list.vue +262 -262
  567. package/src/component/player-vod/video-modal.vue +128 -128
  568. package/src/component/select-options/index.vue +430 -430
  569. package/src/component/select-pages/index.vue +95 -95
  570. package/src/component/svg/index.vue +59 -59
  571. package/src/core/create.js +6 -6
  572. package/src/core/event.js +23 -23
  573. package/src/core/table-methods.js +444 -444
  574. package/src/directive/flexibleResize.js +151 -151
  575. package/src/directive/preventReClick.js +12 -12
  576. package/src/directive/scroll.js +230 -230
  577. package/src/global/variable.js +2 -2
  578. package/src/style/normalize.css +424 -424
  579. package/src/style/style.less +49 -49
  580. package/src/utils/UniRTCv2.js +626 -626
  581. package/src/utils/chatFetch.js +61 -61
  582. package/src/utils/clickoutside.js +75 -75
  583. package/src/utils/crypto.js +25 -25
  584. package/src/utils/index.js +81 -81
  585. package/src/utils/kty-sdk.js +582 -582
  586. package/src/utils/kty.min-1.0.0.js +14378 -14378
  587. package/src/utils/sip-device.js +79 -79
  588. package/src/utils/time-domain.js +249 -249
  589. package/src/utils/trtc.js +1 -1
  590. package/src/utils/utils-map.js +485 -485
  591. package/src/utils/vexutils.js +836 -836
  592. package/.idea/cnhis-design-vue.iml +0 -12
  593. package/.idea/codeStyles/Project.xml +0 -57
  594. package/.idea/codeStyles/codeStyleConfig.xml +0 -5
  595. package/.idea/git_toolbox_prj.xml +0 -15
  596. package/.idea/inspectionProfiles/Project_Default.xml +0 -6
  597. package/.idea/modules.xml +0 -8
  598. package/.idea/vcs.xml +0 -6
@@ -1,1490 +1,1490 @@
1
- <template>
2
- <div class="message-wrapper">
3
- <chat-header :hideHeader="hideHeader" @openPortraitPanel="handleOpenPortraitPanel" v-if="isServer && enable"></chat-header>
4
- <a-layout-content>
5
- <slot></slot>
6
- <div v-if="showBroadcast" class="broadcast-wrap">
7
- <div v-if="broadcastCarousel" class="carousel-item" :class="{ 'has-link': assemblySetting.broadcast.PC.list[0].url }" @click="goToBroadcastLink(assemblySetting.broadcast.PC.list[0])">
8
- <a-icon type="sound" />
9
- <span class="ellips">
10
- {{ assemblySetting.broadcast.PC.list[0].content }}
11
- </span>
12
- </div>
13
- <template v-else>
14
- <a-carousel dotPosition="right" autoplay :dots="false">
15
- <div class="carousel-item" v-for="(item, i) in assemblySetting.broadcast.PC.list" :key="i" :class="{ 'has-link': item.url }" @click="goToBroadcastLink(item)">
16
- <a-icon type="sound" />
17
- <span class="ellips">{{ item.content }}</span>
18
- </div>
19
- </a-carousel>
20
- </template>
21
- </div>
22
- <div v-if="!isHLWYY && sessionHistoryList.length > 1 && !(!onChating && currentTab === 'end') && !sessionEnd" style="position: absolute; top: 0; left: 20px; z-index: 1">
23
- <a-radio-group :value="selectedSession" @change="handleSessionIdChange" buttonStyle="solid">
24
- <a-radio-button :value="item.id + '-' + item.fromId" v-for="item in sessionHistoryList" :key="item.id + item.fromId">{{ item.fromName }}</a-radio-button>
25
- </a-radio-group>
26
- </div>
27
- <div class="online-message" ref="message-wrapper" v-infinite-scroll="handleInfiniteOnLoad" @scroll="scrollEvent">
28
- <template v-for="(item, index) in msgList">
29
- <div :key="index" v-if="item">
30
- <p v-if="isNeedShowTime(item, index)" class="message-time">
31
- {{ formatDate(item.createdTime) }}
32
- </p>
33
- <div v-if="item.recallFlag" style="margin: 10px 0">
34
- <template v-if="handleRecallFlag(item)">
35
- {{ getRecallText(item, isServer) }}
36
- <a style="text-decoration: underline" @click="reEditMessage(item)" v-if="[0, 4].includes(item.content && item.content.type) && onChating && isServer">,{{ i18nText('1.9.368') }}</a>
37
- </template>
38
- <template v-else>
39
- {{ getRecallText(item) || `${item.fromName} ${i18nText('1.9.365')}` }}
40
- </template>
41
- </div>
42
- <!-- 系统消息 -->
43
- <div class="system-msg" v-else-if="isSystemMsg(item)">
44
- <a-icon class="system-icon" theme="filled" type="exclamation-circle" />
45
- <div>
46
- <span v-html="getSystemMsgContent(item)"></span>
47
- <span class="system-btn" v-for="(btn, index) in getTemplateButton(item)" @click="handleTemplateButton(btn)" :key="index">{{ btn.title }}</span>
48
- </div>
49
- </div>
50
- <div
51
- v-else-if="!item.recallFlag"
52
- class="message-item"
53
- :class="[getMessageItemClass(item), { 'read-status': enableReadRecord && item.readStatus }]"
54
- :data-rid="item.readStatus === 'N' ? item.id : ''"
55
- >
56
- <img v-if="!isServer && item.fromPortrait" class="msg-avatar" :src="item.fromPortrait" />
57
-
58
- <div v-else-if="!isServer && item.fromName" class="first-name">
59
- {{ item.fromName && item.fromName.slice(0, 2) }}
60
- </div>
61
- <img v-else-if="!isServer" class="msg-avatar" :src="getAvatar(item)" width="40" height="40" />
62
-
63
- <template v-else>
64
- <img :src="customerStaffIcon" class="msg-avatar" width="40" height="40" v-if="item.fromId === 'SYSTEM'" />
65
-
66
- <img src="../img/system_message.png" class="msg-avatar" width="40" height="40" v-else-if="item.content && item.content.source === 1" />
67
-
68
- <img v-else-if="item.portrait" class="msg-avatar" :src="item.portrait" />
69
-
70
- <div v-else class="first-name">
71
- {{ item.fromName && item.fromName.slice(0, 2) }}
72
- </div>
73
- </template>
74
- <div @contextmenu="handleContextmenu($event, item)" style="display: flex">
75
- <a-spin v-if="item.sending">
76
- <a-icon slot="indicator" type="loading" style="font-size: 24px" spin />
77
- </a-spin>
78
- <a-popconfirm @confirm="handleSendFail(item)" :okText="i18nText('1.1.1.1.3')" :cancelText="i18nText('2.7.1.14')">
79
- <a-icon type="exclamation-circle" slot="icon" />
80
- <div slot="title">
81
- <span>确认重新发送该消息?</span>
82
- </div>
83
- <a-icon v-if="item.fail" theme="filled" type="exclamation-circle" style="font-size: 16px; display: flex; align-items: center; color: red" />
84
- </a-popconfirm>
85
- <div class="content-wrap" v-if="item.content.type === 0">
86
- <span class="right-time">
87
- {{ formatDate(item.createdTime || item.sendTime, true) }}
88
- </span>
89
- <div
90
- :class="{
91
- content: true,
92
- 'has-dictionary': item.dictionaryValues && item.dictionaryValues.length,
93
- 'has-intention': item.intentionList && item.intentionList.length
94
- }"
95
- >
96
- <p v-html="getContentHtml(item)"></p>
97
- <template v-if="item.content.attachments && item.content.attachments.length > 0">
98
- <div v-for="(v, index) in item.content.attachments" :key="index" class="robot-item">
99
- <a v-if="v.type == 1" :href="v.url" target="_blank"> {{ v.desc }}(超链) </a>
100
- <a @click.prevent.stop="$emit('sendQuestion', v, item)" v-if="v.type == 2 || v.type == 3" href="javascript:void(0);">
101
- {{ v.desc }}
102
- </a>
103
- </div>
104
- </template>
105
- <div v-if="item.intentionList && item.intentionList.length" class="intention-wrap">
106
- <div v-for="(v, index) in item.intentionList" :key="index" @click.prevent.stop="$emit('sendIntention', v)">
107
- <svg-icon :icon-class="v.icon || 'moren'" style="font-size: 26px"></svg-icon>
108
- <span>{{ v.name }}</span>
109
- </div>
110
- </div>
111
- </div>
112
- <ul class="dictionary-ul" v-if="item.dictionaryValues && item.dictionaryValues.length">
113
- <li v-for="(v, i) in item.dictionaryValues" :key="i" @click.prevent.stop="$emit('sendDictionary', v, item)">
114
- {{ v }}
115
- </li>
116
- </ul>
117
- <div class="client-btn" v-if="item.autoAccess && item.autoAccess === 'false'">
118
- <svg-icon icon-class="kefu"></svg-icon>
119
- <span @click="$emit('goCustomServe')">转人工客服</span>
120
- </div>
121
- </div>
122
- <div v-else-if="item.content.type === 1" class="upload-image">
123
- <MsgPicture style="max-width: 345px" :image="item.content.content" @click="show(item)" />
124
- </div>
125
- <div class="content-wrap" v-else-if="item.content.type === 2">
126
- <span class="right-time">
127
- {{ formatDate(item.createdTime || item.sendTime, true) }}
128
- </span>
129
- <div class="content">
130
- <img style="width: 16px;margin-right:8px" src="../img/multi-video.png" />
131
- {{ item.content.content }}&nbsp;
132
- <span @click.prevent.stop="reCall(item, 'video')" class="re-call" v-if="item.content.isRefuse == 1">
133
- {{ handleReCall(item) }}
134
- </span>
135
- </div>
136
- </div>
137
- <div class="content-wrap" v-else-if="item.content.type === 5">
138
- <span class="right-time">
139
- {{ formatDate(item.createdTime || item.sendTime, true) }}
140
- </span>
141
- <div class="content">
142
- <img style="width: 16px;margin-right:8px" src="../img/multi-voice.png" />
143
- {{ item.content.content }}&nbsp;
144
- <span @click.prevent.stop="reCall(item, 'voice')" class="re-call" v-if="item.content.isRefuse == 1">
145
- {{ handleReCall(item) }}
146
- </span>
147
- </div>
148
- </div>
149
- <div class="content-wrap" v-else-if="item.content.type === 6">
150
- <span class="right-time">
151
- {{ formatDate(item.createdTime || item.sendTime, true) }}
152
- </span>
153
- <div class="voice-bg content" @click.prevent.stop="playAudio(item)">
154
- <img v-if="item.content.isPlay" src="../img/audio-play.gif" alt="" />
155
- <img v-else src="../img/audio-new.png" alt="" />
156
- <span class="voice-text" :style="AudioStyle(item)"> {{ item.content.duration || 1 }}" </span>
157
- </div>
158
- </div>
159
- <template v-else-if="item.content.type === 4">
160
- <msg-describe v-if="isDescribeMsg(item)" :item="item" />
161
- <div v-else class="content-wrap">
162
- <span class="right-time">
163
- {{ formatDate(item.createdTime || item.sendTime, true) }}
164
- </span>
165
- <MsgPrescription :data="item" :isServer="isServer" :theme="isServer ? '' : 'customer'" @handleTemplateDetail="handleTemplateDetail" @handleTemplateButton="handleTemplateButton" />
166
- </div>
167
- </template>
168
- </div>
169
- <ReadStatus v-if="enableReadRecord" :status="item.readStatus" @click="openReadRecord($event, item)" />
170
- </div>
171
- </div>
172
- </template>
173
- <ul class="menu-main" ref="contextMenu" :style="contextMenuStyle" v-if="contextMenuShow">
174
- <template v-for="(item, index) in rightClickSetting">
175
- <li :key="index" class="menu-item" @click="() => handleRightClick(contextMenuItem, item)">
176
- {{ item.rightClickName }}
177
- </li>
178
- </template>
179
- </ul>
180
- <div class="session-end" v-if="sessionEnd && !isServer">
181
- <span class="end">{{ i18nText('1.9.370') }}</span>
182
- </div>
183
- <a-spin v-if="loading" class="loading-icon" />
184
- </div>
185
- <!-- 图片预览 -->
186
- <viewer class="viewer" ref="viewer" :images="images" @inited="inited" :options="options" style="display: none">
187
- <template slot-scope="scope">
188
- <img v-for="src in scope.images" :src="src" :key="src" />
189
- </template>
190
- </viewer>
191
- <!-- 卡片详情 -->
192
- <a-modal
193
- wrapClassName="chat-footer-modal-small"
194
- :title="modalData.title || i18nText('1.1.1.10')"
195
- :visible="modalShow"
196
- :maskClosable="false"
197
- :mask="true"
198
- width="80%"
199
- destroyOnClose
200
- :okText="i18nText('1.2.1.11.6')"
201
- :footer="null"
202
- @cancel="handleClose"
203
- >
204
- <iframe id="chat-footer-modal" v-if="modalData.targetType === 'LINK_ADDRESS'" width="100%" height="100%" frameborder="0" :src="modalData.address" allow="camera;midi"></iframe>
205
- </a-modal>
206
- </a-layout-content>
207
- <audio style="display: none" controls autoplay ref="audio" preload="auto" crossOrigin="anonymous" media-player="audioPlayer" :src="curAudioUrl" @ended="audioEnd"></audio>
208
- <ReadRecord v-if="readRecordVisible" :style="readRecordStyle" :list="readRecordList" @close="closeReadRecord" />
209
- </div>
210
- </template>
211
-
212
- <script>
213
- import { Layout, Modal, Icon, Spin, Popconfirm, Carousel, Radio } from 'ant-design-vue';
214
- import SvgIcon from '@/component/svg/index.vue';
215
- import InfiniteScroll from '@/directive/scroll';
216
- import { mapGetters, mapMutations, mapActions } from '../store/helper';
217
- import fetch, { qs } from '@/utils/chatFetch';
218
- import MsgDescribe from '../components/msg-describe';
219
- import MsgPicture from '../components/msg-picture';
220
- import MsgPrescription from '../components/msg-prescription';
221
- import ReadStatus from '../components/read-status';
222
- import ReadRecord from '../components/read-record';
223
- import chatHeader from './chatHeader';
224
- import viewerOptions from './mixins/viewerOptions';
225
- import { getRecallText } from '../utils';
226
- import vexutils from '@/utils/vexutils';
227
- import ObserverScroll from '../utils/observer-scroll';
228
-
229
- const customerStaffIcon = require('../img/customer_staff.png');
230
- export default {
231
- mixins: [viewerOptions],
232
- inject: ['store', 'dispatchEvent', 'registerEvent', 'unregisterEvent', 'i18nText'],
233
- props: {
234
- hideHeader: {
235
- type: Boolean,
236
- default: false
237
- },
238
- robotAvatar: {
239
- type: String,
240
- default: ''
241
- },
242
- curChatType: {
243
- type: String
244
- },
245
- isShowPortraitPanel: {
246
- type: Boolean
247
- },
248
- activatedTime: {
249
- type: Number
250
- }
251
- },
252
- data() {
253
- return {
254
- loading: false,
255
- // 卡片详情
256
- modalShow: false,
257
- modalData: {},
258
- // 客户头像
259
- // 右键操作
260
- contextMenuStyle: {
261
- position: 'fixed',
262
- top: 0,
263
- left: 0
264
- },
265
- contextMenuShow: false,
266
- contextMenuItem: null,
267
- // 查看历史消息滚动定位
268
- scrollHeight: 0,
269
- curAudioUrl: '',
270
- lastAudioItem: null,
271
- readRecordStyle: '',
272
- readRecordVisible: false,
273
- readRecordList: []
274
- };
275
- },
276
- computed: {
277
- ...mapGetters([
278
- 'userInfo',
279
- 'currentTab',
280
- 'isAppendMsg',
281
- 'message',
282
- 'msgList',
283
- 'sessionId',
284
- 'serviceId',
285
- 'scrollTo',
286
- 'isServer',
287
- 'clientId',
288
- 'sessionEnd',
289
- 'assemblyId',
290
- 'footerMessage',
291
- 'portraitPanelParams',
292
- 'assemblySetting',
293
- 'clientParams',
294
- 'onChating',
295
- 'classify',
296
- 'collapsed',
297
- 'barStatus',
298
- 'sessionHistoryList',
299
- 'enable',
300
- 'appendList',
301
- 'queueItem',
302
- 'isRecorderVoice',
303
- 'callTimer',
304
- 'requestId'
305
- ]),
306
- // 互联网医院聊天组件
307
- isHLWYY() {
308
- return this.assemblyId === '1227514588661026816'
309
- },
310
- orgId() {
311
- const getters = this.store.getters;
312
- return this.isServer ? this.userInfo?.sysParams?.orgId : getters.accessParams.orgId || getters.queryParams.orgId;
313
- },
314
- userId() {
315
- const getters = this.store.getters;
316
- return this.isServer ? this.userInfo?.sysParams?.userId : getters.accessParams.userId || getters.chatUserId || getters.queryParams.userId;
317
- },
318
- enableReadRecord() {
319
- return this.assemblySetting?.readStatus === 'Y';
320
- },
321
- customerStaffIcon() {
322
- return this.robotAvatar || customerStaffIcon;
323
- },
324
- selectedSession() {
325
- let sessionMap = this.sessionHistoryList.find(item => item.fromId === this.serviceId);
326
- if (sessionMap) {
327
- return sessionMap.id + '-' + sessionMap.fromId;
328
- }
329
- return '';
330
- },
331
- isListClassify() {
332
- return this.assemblySetting.isListClassify === 'Y';
333
- },
334
- showPanel() {
335
- return this.assemblySetting.portraitPanelSetting?.targetIs;
336
- },
337
- showBroadcast() {
338
- let res = (!this.isServer && this.assemblySetting?.broadcast?.PC?.isChecked == 'Y' && this.assemblySetting?.broadcast?.PC?.list?.length > 0) || false;
339
- return res;
340
- },
341
- broadcastCarousel() {
342
- return (this.assemblySetting.broadcast && this.assemblySetting.broadcast.PC.list.length == 1) || false;
343
- },
344
- rightClickSetting() {
345
- const { isServer } = this;
346
- if (!isServer) return [];
347
- const isSelf = this.contextMenuItem.fromId === this.serviceId;
348
- const setting = this.assemblySetting?.rightClickSetting || [];
349
- const doArr = ['resend', 'rollBack'];
350
- return setting.filter(item => {
351
- const { rightClickDo } = item;
352
- return isSelf || doArr.indexOf(rightClickDo) < 0;
353
- });
354
- },
355
- handleReCall() {
356
- return function(item) {
357
- if (item.content?.source === 1) {
358
- return '[点击回拨]';
359
- } else if (item.senderFlag == 1) {
360
- return `[点击${this.isServer ? '回拨' : '重拨'}]`;
361
- }
362
- if (item.senderFlag == 2) {
363
- return `[点击${this.isServer ? '重拨' : '回拨'}]`;
364
- }
365
- if (this.isServer) {
366
- let isRight = item.fromId !== 'SYSTEM' && item.fromId && item.fromId === this.serviceId;
367
- return `[点击${isRight ? '重拨' : '回拨'}]`;
368
- } else {
369
- let isRight = (item.fromId !== 'SYSTEM' && item.toId === 'SYSTEM') || item.fromId === this.clientId || (item.mode === 'bot' && !item.fromId);
370
- return `[点击${isRight ? '重拨' : '回拨'}]`;
371
- }
372
- };
373
- },
374
- handleRecallFlag() {
375
- return function(item) {
376
- return this.serviceId == item.fromId;
377
- };
378
- },
379
- AudioStyle() {
380
- return function(item) {
381
- let { duration = 1 } = item.content;
382
- if (!duration) {
383
- duration = 1;
384
- }
385
- let curWidth = (+duration / 60) * 400;
386
- return {
387
- width: curWidth < 13 ? 13 : curWidth + 'px'
388
- };
389
- };
390
- },
391
- showTimeDuration() {
392
- let { variableSetting = null } = this.assemblySetting || {};
393
- if (!variableSetting) return 3;
394
- let matchItem = variableSetting.find(v => v.name === 'CHAT_SEPARATION_TIME');
395
- if (!matchItem) return 3;
396
- if (!+matchItem.value) return 3;
397
- return +matchItem.value;
398
- },
399
- images() {
400
- return this.msgList.filter(v => !v.recallFlag && v.content.type == 1)?.map(v => v.content?.content);
401
- }
402
- },
403
- created() {
404
- this.attachEvent();
405
- this.getUserInformation();
406
- },
407
- methods: {
408
- ...mapMutations([
409
- 'setIsAppendMsg',
410
- 'setMsgList',
411
- 'setScrollTo',
412
- 'setFooterMessage',
413
- 'setIsScrollListChange',
414
- 'setListChangeItem',
415
- 'setSessionId',
416
- 'setOnChating',
417
- 'setServiceId',
418
- 'setShowAudio',
419
- 'setVideoMode',
420
- 'setVideoMembers',
421
- 'setIsRecorderVoice',
422
- 'setDoctorInfo',
423
- 'setCallTimer'
424
- ]),
425
- ...mapActions({
426
- getEarlierMsg: 'setMsgList',
427
- handleBotChat: 'handleBotChat',
428
- sendMessage: 'sendMessage'
429
- }),
430
- // 获取医生信息
431
- getUserInformation() {
432
- const url = '/user/getUserInformation';
433
- fetch.get(url).then(({ data }) => {
434
- if (data.result === 'SUCCESS') {
435
- this.setDoctorInfo(data?.map?.sysUserInformation)
436
- console.log('data?.map?.sysUserInformation', data?.map?.sysUserInformation);
437
- }
438
- });
439
- },
440
- getMessageItemClass(item) {
441
- // content.source === 1, 代表为系统消息, 系统统一左侧展示
442
- if (item.content?.source === 1) {
443
- return {
444
- left: true
445
- };
446
- }
447
-
448
- // 服务端 senderFlag 1 左 2 右
449
- // 客户端 senderFlag 1 右 2 左
450
- if (item.content?.source === 1) {
451
- return {
452
- left: true
453
- };
454
- } else if (item.senderFlag == 1) {
455
- if (this.isServer) {
456
- return {
457
- left: true
458
- };
459
- } else {
460
- return {
461
- right: true
462
- };
463
- }
464
- }
465
- if (item.senderFlag == 2) {
466
- if (this.isServer) {
467
- return {
468
- right: true
469
- };
470
- } else {
471
- return {
472
- left: true
473
- };
474
- }
475
- }
476
- if (this.isServer) {
477
- return {
478
- right: item.fromId !== 'SYSTEM' && item.fromId && item.fromId === this.serviceId,
479
- left: item.fromId === 'SYSTEM' || item.fromId === this.clientId || item.content?.source === 1 || !item.fromId
480
- };
481
- } else {
482
- let isRight, isLeft;
483
- if (this.curChatType == 'robot') {
484
- isRight = item.fromId !== 'SYSTEM';
485
- isLeft = item.fromId === 'SYSTEM';
486
- } else {
487
- isRight = (item.fromId !== 'SYSTEM' && item.toId === 'SYSTEM') || item.fromId === this.clientId || (item.mode === 'bot' && !item.fromId);
488
- isLeft = item.fromId === 'SYSTEM' || item.fromId !== this.clientId || item.content?.source === 1;
489
- }
490
- return {
491
- right: isRight,
492
- left: isLeft
493
- };
494
- }
495
- },
496
- /* 向上滚动刷新 */
497
- handleInfiniteOnLoad() {
498
- if (this.curChatType === 'robot') return;
499
- this.loading = true;
500
- this.getEarlierMsg().then(msg => {
501
- if (msg) {
502
- this.$message.info(msg);
503
- }
504
- this.loading = false;
505
- });
506
- },
507
- show(item) {
508
- const src = item.content.content;
509
- // this.images = vexutils.imgs2imgArr(src);
510
- this.$viewer.view(this.images.findIndex(v => v === src));
511
- this.$viewer.show();
512
- this.readSingleMsg(item);
513
- },
514
- /* 滚动条定位 */
515
- handleScrollTo(scrollTo) {
516
- let target = this.$refs['message-wrapper'];
517
- this.$nextTick().then(() => {
518
- if (!target) return;
519
- let top = target.scrollHeight;
520
- if (scrollTo === 'current') {
521
- top = target.scrollHeight - this.scrollHeight;
522
- }
523
- if (typeof scrollTo === 'number') {
524
- top = scrollTo;
525
- }
526
- target.scrollTo({
527
- left: 0,
528
- top,
529
- behavior: 'instant'
530
- });
531
- this.scrollHeight = 0;
532
- });
533
- this.setScrollTo('');
534
- },
535
-
536
- /**
537
- * 是否需要显示时间
538
- */
539
- isNeedShowTime(item, index) {
540
- const curMsgTime = item.createdTime || new Date().getTime();
541
- let duration = 60 * 1000 * this.showTimeDuration;
542
- let lastMsgTime;
543
- if (index == 0) {
544
- lastMsgTime = 0;
545
- } else {
546
- lastMsgTime = this.msgList[index - 1].createdTime;
547
- }
548
- if (Number(curMsgTime) - Number(lastMsgTime) > duration) {
549
- return true;
550
- } else {
551
- return false;
552
- }
553
- },
554
- formatDate(date) {
555
- return vexutils.formatDate(date, true);
556
- },
557
- getAvatar(item) {
558
- if (item.content && item.content.source === 1) {
559
- return require('../img/system_message.png');
560
- }
561
- if (item.mode === 'bot' && !item.fromId) {
562
- return require('../img/default.png');
563
- }
564
- if (this.isServer) {
565
- if (item.fromId === this.clientId || !item.fromId) {
566
- return require('../img/default.png');
567
- } else {
568
- return this.customerStaffIcon;
569
- }
570
- } else {
571
- if (this.curChatType == 'robot') {
572
- if (item.fromId === 'SYSTEM') {
573
- return this.customerStaffIcon;
574
- } else {
575
- return require('../img/default.png');
576
- }
577
- } else {
578
- if (item.fromId !== this.clientId) {
579
- return this.customerStaffIcon;
580
- } else {
581
- return require('../img/default.png');
582
- }
583
- }
584
- }
585
- },
586
- getContent(item) {
587
- let content;
588
- if (vexutils.isJSON(item.content.content)) {
589
- content = JSON.parse(item.content.content);
590
- } else {
591
- content = item.content.content;
592
- }
593
- return content;
594
- },
595
- getContentHtml(item) {
596
- return !~item.content.content.indexOf('emoji') ? this.$xss(item.content.content) : item.content.content;
597
- },
598
- getTemplateTitle(item) {
599
- return this.getContent(item)?.title;
600
- },
601
- showArrow(item) {
602
- let setting = this.getContent(item)?.customerSetting || {};
603
- if (this.isServer) {
604
- setting = this.getContent(item)?.serverSetting || {};
605
- }
606
- let { address } = setting;
607
- if (!address) return false;
608
- return true;
609
- },
610
- getTemplateTitleIcon(item) {
611
- return this.getContent(item)?.icon;
612
- },
613
- getTemplateContent(item) {
614
- let content = this.getContent(item);
615
- let array = [];
616
- try {
617
- array = content.content.split('##');
618
- } catch {
619
- console.log(content, 'error');
620
- }
621
- array = array.map(item => `<div>${item}</div>`);
622
- return array.join('');
623
- },
624
- getSystemMsgContent(item) {
625
- let content = this.getContent(item);
626
- let array = [];
627
- try {
628
- array = content.content.split('##');
629
- } catch {
630
- console.log(content, 'error');
631
- }
632
- return array.join('<br>');
633
- },
634
- getTemplateButton(item) {
635
- let content = this.getContent(item);
636
- const setting = this.isServer ? content.serverSetting : content.customerSetting;
637
- return setting?.toolbar_button || [];
638
- },
639
- // 处理模板按钮,详情等跳转
640
- handleTemplate(data, dispatch = true) {
641
- let { targetType, address, params = [], openMode } = data;
642
- dispatch &&
643
- this.dispatchEvent('click_templateCard', {
644
- ...data
645
- });
646
- if (!address) return;
647
- if (targetType === 'LINK_ADDRESS') {
648
- let urlParams = [];
649
- params.forEach(({ p_name, p_value }) => {
650
- urlParams.push(`${p_name}=${p_value}`);
651
- });
652
- if (address.includes('?')) {
653
- address += `&${urlParams.join('&')}`;
654
- } else {
655
- address += `?${urlParams.join('&')}`;
656
- }
657
- if (openMode === 'WINDOW') {
658
- window.open(address);
659
- return;
660
- }
661
- if (openMode === 'EJECT') {
662
- // 打开云his页面需要从cookie内获取最新的jsessionids
663
- if (address.includes('jsessionidsFlag=1')) {
664
- let jsessionids = vexutils.cookie.get('jsessionids')
665
- address = address + `&jsessionids=${jsessionids}`
666
- }
667
- this.modalShow = true;
668
- this.modalData = Object.assign({}, data, {
669
- address
670
- });
671
- this.$nextTick().then(() => {
672
- window.addEventListener('message', this.iframeEvent);
673
- });
674
- return;
675
- }
676
- }
677
- },
678
- iframeEvent(event) {
679
- const method = event?.data?.method;
680
- this[method]?.(event);
681
- },
682
- // 发送
683
- handleParentMessageSend({ data, source, origin }) {
684
- let params = {
685
- assemblyId: this.assemblyId,
686
- sessionId: this.sessionId,
687
- orgId: this.orgId
688
- };
689
- params = Object.assign(params, data.data);
690
- return fetch.post('/chat/access/sendToolBarData', qs.stringify(params)).then(({ data }) => {
691
- if (data.result === 'SUCCESS') {
692
- source.postMessage({ status: 0, resultMsg: data.resultMsg }, origin);
693
- } else {
694
- source.postMessage({ status: 1, resultMsg: data.resultMsg }, origin);
695
- }
696
- });
697
- },
698
- // 关闭
699
- handleClose() {
700
- this.modalShow = false;
701
- window.removeEventListener('message', this.iframeEvent);
702
- },
703
- handleTemplateDetail(item) {
704
- let content = this.getContent(item) || {};
705
- let setting = content.customerSetting || {};
706
- if (this.isServer) {
707
- setting = content.serverSetting || {};
708
- }
709
- this.readSingleMsg(item);
710
- this.handleTemplate(setting);
711
- },
712
- handleTemplateButton(btn) {
713
- this.handleTemplate(btn);
714
- },
715
- // 消息撤回
716
- handleRecall(item) {
717
- let params = {
718
- assemblyId: this.assemblyId,
719
- sessionId: this.sessionId,
720
- messageId: item.id,
721
- messageType: item.content?.type,
722
- userId: this.userInfo?.sysParams?.userId || ''
723
- };
724
-
725
- fetch.post('/chat/service/recallMessage', qs.stringify(params)).then(({ data }) => {
726
- if (data.result === 'SUCCESS') {
727
- this.dispatchEvent('msg_recallSuccess', {
728
- message: { ...item },
729
- ...params
730
- });
731
- this.$set(item, 'recallFlag', 1);
732
- // 撤回成功提示
733
- let title = this.getContent(item)?.title;
734
- let text = title ? `已撤回[${title}]` : '已撤回';
735
- this.$message.success(text);
736
-
737
- if (!this.isLastMessage(item.id)) return;
738
- this.setIsScrollListChange(true);
739
- this.setListChangeItem({
740
- opt: 1,
741
- from: this.clientId,
742
- sessionId: this.sessionId,
743
- msgCount: 0,
744
- lastContentRecallFlag: 1,
745
- lastContent: vexutils.clone(item.content, true),
746
- lastTime: new Date().getTime()
747
- });
748
- } else {
749
- this.$message.error(data.resultMsg);
750
- }
751
- });
752
- },
753
- handleJumpUrl(contextMenuItem, item) {
754
- const title = item.rightClickName;
755
- const url = item.rightClickUrl.replace(/{(.*?)}/g, (match, p) => {
756
- return eval(p);
757
- });
758
- this.dispatchEvent('open_jumpUrl', {
759
- title: title,
760
- url: url
761
- });
762
- },
763
- // 右键菜单功能
764
- handleRightClick(contextMenuItem, item) {
765
- if (item.rightClickDo === 'rollBack') {
766
- this.handleRecall(contextMenuItem);
767
- } else if (item.rightClickDo === 'jumpUrl') {
768
- this.handleJumpUrl(contextMenuItem, item);
769
- } else if (item.rightClickDo === 'resend') {
770
- this.resendMessage(contextMenuItem);
771
- }
772
- },
773
- resendMessage(item) {
774
- fetch({
775
- method: 'POST',
776
- url: '/chat/service/resendMessage',
777
- data: qs.stringify({
778
- assemblyId: this.assemblyId,
779
- sessionId: this.sessionId,
780
- messageId: item.id
781
- })
782
- });
783
- },
784
- // 重新编辑
785
- reEditMessage(item) {
786
- if (item.content?.type === 4) {
787
- let content = this.getContent(item);
788
- this.handleTemplate(content.serverSetting.reEditSetting, false);
789
- return;
790
- }
791
- this.setFooterMessage(this.footerMessage + item.content.content.slice(5, -6));
792
- },
793
- observeMsg() {
794
- if (this.enableReadRecord) {
795
- if (!this._observer) {
796
- this._observer = new ObserverScroll(this.$refs['message-wrapper'], this.readMsg);
797
- this.$on('hook:beforeDestroy', () => {
798
- this._observer.remove();
799
- this._observer = null;
800
- });
801
- }
802
- this._observer.disconnect();
803
- if (this.msgList.length) {
804
- this.$nextTick(() => {
805
- const el = this.$refs['message-wrapper'];
806
- const els = el?.querySelectorAll('.message-item.left');
807
- els && this._observer.observe(Array.from(els).filter(item => item.getAttribute('data-rid')));
808
- });
809
- }
810
- }
811
- },
812
- readMsg(ids, enableVaid = true, notify = true, post = true) {
813
- const msgList = this.msgList;
814
- const idKeys = msgList.reduce((obj, item, i) => {
815
- const id = item?.id;
816
- if (id) obj[id] = i;
817
- return obj;
818
- }, {});
819
- const data = [];
820
- const validFn = (item, type) => {
821
- /**
822
- * 针对文本、表情消息类型,只要在聊天区域内,视为已读,
823
- * 图片、语音类型消息,需要点击查看后,才视为已读,
824
- * 模板 类型消息,如果没有详情链接,只要在聊天区域内,视为已读,若有详情链接,需要点击查看后,才视为已读
825
- */
826
- const jumpType = `,6,1,`;
827
- if (jumpType.includes(type)) return false;
828
- if (type == 4 && !this.isDescribeMsg(item)) {
829
- let content = this.getContent(item) || {};
830
- let setting = (this.isServer ? content.serverSetting : content.customerSetting) || {};
831
- return !setting?.address;
832
- }
833
- return true;
834
- };
835
- const value = ids.filter(id => {
836
- const i = idKeys[id];
837
- let item = msgList[i];
838
- if (item) {
839
- const type = item.content.type;
840
- if (!enableVaid || validFn(item, type)) {
841
- item.readStatus = 'Y';
842
- data.push({ id, type });
843
- return true;
844
- }
845
- return false;
846
- }
847
- });
848
- if (!value.length) return;
849
- notify &&
850
- this.dispatchEvent('msg_readed', {
851
- data,
852
- assemblyId: this.assemblyId,
853
- sessionId: this.sessionId
854
- });
855
- post && this.postReadMessage({ ids: value.join(',') });
856
- },
857
- postReadMessage(data) {
858
- fetch.post(
859
- `/chat/${this.isServer ? 'service' : 'access'}/readMessage`,
860
- qs.stringify({
861
- assemblyId: this.assemblyId,
862
- orgId: this.orgId,
863
- userId: this.userId,
864
- sessionId: this.sessionId,
865
- ...data
866
- })
867
- );
868
- },
869
- readSingleMsg(item) {
870
- if (this.enableReadRecord && item.readStatus === 'N') {
871
- const cls = this.getMessageItemClass(item);
872
- if (!cls?.left) return;
873
- this.readMsg([item.id], false);
874
- }
875
- },
876
- closeReadRecord() {
877
- this.readRecordVisible = false;
878
- this.readRecordList = [];
879
- },
880
- openReadRecord(event, item) {
881
- const readStatus = item.readStatus;
882
- if (readStatus) {
883
- const cls = this.getMessageItemClass(item);
884
- if (!cls.right) return;
885
- const rect = event.target.getBoundingClientRect();
886
- const w = 440;
887
- const h = 340;
888
- const top = Math.max(0, Math.min(document.body.clientHeight - h, rect.top - h / 2));
889
- const left = Math.max(0, Math.min(document.body.clientWidth - w, rect.left - w));
890
- fetch({
891
- url: `/chat/${this.isServer ? 'service' : 'access'}/getMessageMember`,
892
- method: 'get',
893
- params: {
894
- assemblyId: this.assemblyId,
895
- messageId: item.id,
896
- orgId: this.isServer ? undefined : this.orgId
897
- }
898
- }).then(res => {
899
- const list = res.data?.list || [];
900
- this.readRecordStyle = `top:${top}px;left:${left}px;position:fixed;z-index:10`;
901
- this.readRecordVisible = list.length > 0;
902
- this.readRecordList = Object.freeze(list);
903
- if (item.readStatus && list.length) {
904
- const unread = list.filter(item => item.readStatus == 'N').length;
905
- const single = list.length < 2;
906
- const status = unread > 0 ? (single ? 'N' : `${unread}`) : single ? 'Y' : 'ALL';
907
- // 检查是否更新人数
908
- if (item.readStatus != status) {
909
- item.readStatus = status;
910
- }
911
- }
912
- });
913
- }
914
- },
915
- isLastMessage(id) {
916
- return this.msgList.findIndex(msg => msg.id === id) === this.msgList.length - 1;
917
- },
918
- isSystemMsg(item) {
919
- return item.content.type == 4 && this.getContent(item).styleId == '2';
920
- },
921
- isDescribeMsg(item) {
922
- return this.getContent(item).styleId == '3';
923
- },
924
- async handleOpenPortraitPanel(item) {
925
- if (!this.showPanel || item.fromId === this.serviceId || !this.isServer || !this.isShowPortraitPanel) return;
926
- const { portraitPanelParams, clientParams } = this;
927
- const panelSetting = this.assemblySetting?.portraitPanelSetting;
928
- this.dispatchEvent('open_portraitPanel', {
929
- panelSetting,
930
- clientParams,
931
- panelParams: portraitPanelParams,
932
- targetId: panelSetting?.targetId,
933
- targetType: panelSetting?.targetType
934
- });
935
- },
936
- handleContextmenu(e, item) {
937
- // 系统消息,客户消息,待接入消息,历史消息不允许撤回
938
- // 模板-描述消息禁用右键
939
- if (item.content?.source === 1 || !this.sessionId || (this.sessionId && !this.onChating) || (item.content.type === 4 && this.isDescribeMsg(item))) return;
940
- this.contextMenuItem = item;
941
- if (this.rightClickSetting.length < 1) return;
942
- e.preventDefault();
943
- let left = e.pageX + 'px';
944
- let top = e.pageY + 'px';
945
- this.contextMenuStyle = Object.assign(this.contextMenuStyle, {
946
- top,
947
- left
948
- });
949
- this.contextMenuShow = true;
950
- },
951
- scrollEvent(e) {
952
- console.log('e.target.scrollTop->', e.target.scrollTop);
953
- this._scrollTop = e.target.scrollTop;
954
- },
955
- /* 事件绑定 */
956
- attachEvent() {
957
- this.scrollEvent = vexutils.debounce(this.scrollEvent, 800);
958
- const handleClick = () => {
959
- this.contextMenuShow = false;
960
- };
961
- const handleRasize = () => {
962
- if (this.readRecordVisible) {
963
- let style = this.readRecordStyle;
964
- const top = style.match(topReg)?.[1];
965
- const topReg = /top:([\d.]+)/;
966
- const maxH = document.body.clientHeight - 360;
967
- if (top && maxH < top) {
968
- style = style.replace(topReg, `top:${maxH}`);
969
- }
970
- const leftReg = /left:([\d.]+)/;
971
- const left = style.match(leftReg)?.[1];
972
- const maxW = document.body.clientWidth - 440;
973
- if (left && maxW < left) {
974
- style = style.replace(leftReg, `left:${maxW}`);
975
- }
976
- this.readRecordStyle = style;
977
- }
978
- };
979
- window.addEventListener('resize', handleRasize);
980
- document.addEventListener('click', handleClick);
981
- this.registerEvent('openPortraitPanel', this.handleOpenPortraitPanel);
982
- const readMessage = data => {
983
- this.readMsg(data.ids.split(','), false, false, false);
984
- this.postReadMessage(data);
985
- };
986
- this.registerEvent('readMessage', readMessage);
987
- this.$on('hook:beforeDestroy', () => {
988
- window.removeEventListener('resize', handleRasize);
989
- document.removeEventListener('click', handleClick);
990
- this.unregisterEvent('openPortraitPanel', this.handleOpenPortraitPanel);
991
- this.unregisterEvent('readMessage', readMessage);
992
- });
993
- },
994
- goToBroadcastLink(item) {
995
- if (!item.url) return;
996
- window.open(item.url);
997
- },
998
- /**
999
- * 客服切换
1000
- */
1001
- handleSessionIdChange(e) {
1002
- let [sessionId, serviceId] = e.target.value.split('-');
1003
- this.setSessionId(sessionId);
1004
- this.setServiceId(serviceId);
1005
- let isOnChating = this.sessionHistoryList.find(item => item.fromId === serviceId).onChating;
1006
- this.setOnChating(!!isOnChating);
1007
- this.setMsgList([]);
1008
- this.getEarlierMsg();
1009
- },
1010
- handleSendFail(item) {
1011
- this.sendMessage(Object.assign(item.content, { failItem: item }));
1012
- },
1013
- reCall(type) {
1014
- this.setShowAudio(true);
1015
- this.setVideoMode(type === 'voice' ? 1 : 2);
1016
- this.$nextTick(() => {
1017
- this.setVideoMembers([]);
1018
- });
1019
- },
1020
- playAudio(item) {
1021
- let { content = '' } = item?.content || {};
1022
- if (!content) return;
1023
- if (this.lastAudioItem && (item.id !== this.lastAudioItem.id || item.createdTime !== this.lastAudioItem.createdTime)) {
1024
- this.$set(this.lastAudioItem.content, 'isPlay', false);
1025
- }
1026
-
1027
- this.readSingleMsg(item);
1028
- this.lastAudioItem = item;
1029
- this.$set(item.content, 'isPlay', !item.content.isPlay);
1030
- this.$nextTick(() => {
1031
- this.curAudioUrl = content;
1032
- let audioRef = this.$refs['audio'];
1033
- if (!audioRef) return;
1034
- audioRef.load();
1035
- setTimeout(() => {
1036
- if (item.content.isPlay) {
1037
- let playPromise = audioRef.play();
1038
- if (playPromise !== undefined) {
1039
- playPromise.then(() => {
1040
- audioRef.play();
1041
- });
1042
- }
1043
- } else {
1044
- audioRef.pause();
1045
- this.curAudioUrl = '';
1046
- }
1047
- });
1048
- });
1049
- },
1050
- audioEnd() {
1051
- if (!this.lastAudioItem) return;
1052
- this.$set(this.lastAudioItem.content, 'isPlay', false);
1053
- },
1054
- getRecallText(item, flag = false) {
1055
- return getRecallText(this, item, flag);
1056
- }
1057
- },
1058
- directives: {
1059
- InfiniteScroll
1060
- },
1061
- components: {
1062
- AModal: Modal,
1063
- [Layout.Content.name]: Layout.Content,
1064
- [Spin.name]: Spin,
1065
- [Icon.name]: Icon,
1066
- [Popconfirm.name]: Popconfirm,
1067
- [Radio.Button.name]: Radio.Button,
1068
- [Radio.Group.name]: Radio.Group,
1069
- [Carousel.name]: Carousel,
1070
- SvgIcon,
1071
- MsgPrescription,
1072
- ReadStatus,
1073
- ReadRecord,
1074
- MsgPicture,
1075
- MsgDescribe,
1076
- chatHeader
1077
- },
1078
- watch: {
1079
- msgList() {
1080
- this.observeMsg();
1081
- },
1082
- isAppendMsg: {
1083
- immediate: true,
1084
- handler(isAppendMsg) {
1085
- if (isAppendMsg) {
1086
- let newArr = [];
1087
- if (vexutils.isPlainObject(this.message)) {
1088
- newArr = [...this.msgList, this.message];
1089
- } else if (vexutils.isArray(this.message)) {
1090
- newArr = [...this.msgList, ...this.message];
1091
- }
1092
- this.setMsgList(newArr);
1093
- this.setIsAppendMsg(false);
1094
- this.setScrollTo('bottom');
1095
- }
1096
- }
1097
- },
1098
- scrollTo: {
1099
- immediate: true,
1100
- handler(scrollTo) {
1101
- if (scrollTo) {
1102
- this.handleScrollTo(scrollTo);
1103
- }
1104
- }
1105
- },
1106
- activatedTime: {
1107
- immediate: true,
1108
- handler(value) {
1109
- this.handleScrollTo(this._scrollTop);
1110
- }
1111
- },
1112
- /**
1113
- * 图片懒加载处理
1114
- */
1115
- appendList: {
1116
- handler() {
1117
- if (!this.scrollHeight) {
1118
- let target = this.$refs['message-wrapper'];
1119
- this.scrollHeight = target.scrollHeight;
1120
- }
1121
- if (this.appendList.length === this.msgList.length || this.appendList.length === 1) {
1122
- this.setScrollTo('bottom');
1123
- } else {
1124
- this.setScrollTo('current');
1125
- }
1126
- }
1127
- },
1128
- isRecorderVoice: {
1129
- immediate: true,
1130
- handler(val) {
1131
- if (!val) return;
1132
- this.curAudioUrl = '';
1133
- this.setIsRecorderVoice(false);
1134
- if (!this.lastAudioItem) return;
1135
- this.$set(this.lastAudioItem.content, 'isPlay', false);
1136
- }
1137
- }
1138
- }
1139
- };
1140
- </script>
1141
-
1142
- <style lang="less" scoped>
1143
- @import '../style/emoji.css';
1144
- @import '../style/message.mixin.less';
1145
- .message-wrapper {
1146
- .createMessageStyle(#ebebeb, @primary-color);
1147
- flex: 1;
1148
- height: 0;
1149
- // position: relative;
1150
- display: flex;
1151
- flex-direction: column;
1152
- overflow-x: hidden;
1153
- .broadcast-wrap {
1154
- height: 30px;
1155
- text-align: center;
1156
- font-size: 14px;
1157
- background-color: #fffbe8;
1158
- /deep/ .carousel-item {
1159
- display: inline-flex !important;
1160
- justify-content: center;
1161
- align-items: center;
1162
- width: 100%;
1163
- height: 30px;
1164
- text-align: center;
1165
- &.has-link {
1166
- color: #ed6a0c;
1167
- cursor: pointer;
1168
- }
1169
- span {
1170
- display: inline-block;
1171
- max-width: calc(100% - 30px);
1172
- margin-left: 4px;
1173
- }
1174
- }
1175
- }
1176
-
1177
- .loading-icon {
1178
- position: absolute;
1179
- top: 0;
1180
- left: 50%;
1181
- }
1182
- }
1183
- /deep/ .ant-layout-content {
1184
- flex: 1;
1185
- display: flex;
1186
- flex-direction: column;
1187
- position: relative;
1188
- }
1189
- .online-message {
1190
- flex: 1;
1191
- height: 100%;
1192
- // height: 608px;
1193
- overflow-y: auto;
1194
- overflow-x: hidden;
1195
- padding: 10px 22px;
1196
- // margin: 10px 0;
1197
- text-align: center;
1198
- &::-webkit-scrollbar {
1199
- width: 5px;
1200
- }
1201
- .message-time {
1202
- display: inline-block;
1203
- color: #666;
1204
- font-size: 14px;
1205
- border-radius: 4px;
1206
- padding: 4px;
1207
- margin: 0 auto;
1208
- margin-bottom: 16px;
1209
- }
1210
- > div {
1211
- margin-bottom: 16px;
1212
- }
1213
- .message-item {
1214
- display: flex;
1215
- /deep/ .chat-read-status-text {
1216
- display: none;
1217
- }
1218
- &.right.read-status {
1219
- padding-bottom: 24px;
1220
- position: relative;
1221
- /deep/ .chat-read-status-text {
1222
- display: block;
1223
- position: absolute;
1224
- right: 50px;
1225
- bottom: 0;
1226
- }
1227
- }
1228
- .right-time {
1229
- display: none;
1230
- position: absolute;
1231
- top: -16px;
1232
- left: 10px;
1233
- font-size: 12px;
1234
- color: #b0b1b1;
1235
- white-space: nowrap;
1236
- }
1237
-
1238
- .content-wrap {
1239
- padding: 0 10px;
1240
- text-align: left;
1241
- position: relative;
1242
- &:hover {
1243
- .right-time {
1244
- display: block;
1245
- }
1246
- }
1247
-
1248
- .robot-item {
1249
- text-align: left;
1250
- a {
1251
- text-decoration: underline;
1252
- }
1253
- }
1254
- .intention-wrap {
1255
- display: flex;
1256
- flex-wrap: wrap;
1257
- padding: 8px 0 4px 0;
1258
- background-color: #fff;
1259
- border-radius: 0 0 8px 8px;
1260
- > div {
1261
- display: inline-flex;
1262
- flex-direction: column;
1263
- align-items: center;
1264
- width: 60px;
1265
- height: 54px;
1266
- cursor: pointer;
1267
-
1268
- &:hover {
1269
- color: #5585f5;
1270
- span {
1271
- color: #5585f5;
1272
- }
1273
- }
1274
- }
1275
- span {
1276
- margin-top: 8px;
1277
- max-width: 90%;
1278
- text-overflow: ellipsis;
1279
- overflow: hidden;
1280
- white-space: nowrap;
1281
- font-size: 12px;
1282
- color: #6a6a6a;
1283
- }
1284
- }
1285
- .dictionary-ul {
1286
- padding-top: 8px;
1287
- display: flex;
1288
- flex-wrap: wrap;
1289
- li {
1290
- padding: 4px 16px;
1291
- margin-right: 16px;
1292
- font-size: 14px;
1293
- color: #5585f5;
1294
- background: #efefef;
1295
- border-radius: 16px;
1296
- cursor: pointer;
1297
- }
1298
- }
1299
- .client-btn {
1300
- display: inline-block;
1301
- padding: 4px 8px;
1302
- margin-top: 8px;
1303
- color: #fff;
1304
- font-size: 12px;
1305
- border-radius: 20px;
1306
- background: #5585f5;
1307
- cursor: pointer;
1308
- }
1309
- .re-call {
1310
- cursor: pointer;
1311
- color: #5585f5;
1312
- }
1313
- .voice-bg {
1314
- display: flex;
1315
- align-items: center;
1316
- cursor: pointer;
1317
- img {
1318
- mix-blend-mode: difference;
1319
- display: inline-block;
1320
- width: 14px;
1321
- }
1322
- .voice-text {
1323
- display: inline-block;
1324
- }
1325
- span {
1326
- margin-left: 8px;
1327
- }
1328
- }
1329
- }
1330
- .has-dictionary {
1331
- padding: 0;
1332
- text-align: left;
1333
- background: unset;
1334
- }
1335
- .content {
1336
- padding: 10px;
1337
- background: #ebebeb;
1338
- border-radius: 0px 8px 8px 8px;
1339
- max-width: 500px;
1340
- margin: 0px;
1341
- line-height: 20px;
1342
- font-size: 14px;
1343
- color: #000;
1344
- text-align: left;
1345
- word-break: break-word;
1346
- /deep/ p {
1347
- margin: 0;
1348
- }
1349
- }
1350
- .upload-image {
1351
- margin-left: 8px;
1352
- }
1353
- &.right {
1354
- flex-direction: row-reverse;
1355
- .right-time {
1356
- left: unset;
1357
- right: 10px;
1358
- }
1359
- .content-wrap {
1360
- text-align: right;
1361
- .robot-item {
1362
- text-align: right;
1363
- }
1364
- .voice-bg {
1365
- flex-direction: row-reverse;
1366
- align-self: flex-end;
1367
- img {
1368
- transform: rotateY(180deg);
1369
- }
1370
- span {
1371
- margin-left: 0;
1372
- margin-right: 8px;
1373
- text-align: right;
1374
- }
1375
- }
1376
- }
1377
- .content-wrap .content {
1378
- background-color: #e7f0ff;
1379
- border-radius: 8px 0px 8px 8px;
1380
- border: 1px solid #e7f0ff;
1381
- box-sizing: border-box;
1382
- }
1383
- .upload-image {
1384
- margin-right: 8px;
1385
- }
1386
- }
1387
- }
1388
- .bot-message {
1389
- display: block;
1390
- text-decoration: underline;
1391
- color: #5585f5;
1392
- cursor: pointer;
1393
- }
1394
- .session-end {
1395
- display: flex;
1396
- justify-content: center;
1397
- margin-top: 15px;
1398
- margin-bottom: 25px;
1399
- .end {
1400
- display: flex;
1401
- justify-content: center;
1402
- width: 30%;
1403
- height: 36px;
1404
- line-height: 36px;
1405
- background-color: #e8e8e8;
1406
- color: #939393;
1407
- font-size: 14px;
1408
- border-radius: 18px;
1409
- }
1410
- }
1411
- .menu-tip {
1412
- font-size: 20px;
1413
- align-self: flex-end;
1414
- margin-bottom: 8px;
1415
- background: #f6f6f6;
1416
- height: 25px;
1417
- line-height: 25px;
1418
- width: 30px;
1419
- border-radius: 4px;
1420
- }
1421
- .menu-item {
1422
- :hover {
1423
- background: red;
1424
- }
1425
- }
1426
- .menu-main {
1427
- width: 100px;
1428
- background-color: #101627;
1429
- color: #fff;
1430
- text-align: left;
1431
- border-radius: 6px;
1432
- padding: 10px 0;
1433
- z-index: 3000;
1434
- cursor: pointer;
1435
- .menu-item {
1436
- padding: 0 10px;
1437
- }
1438
- .menu-item:hover {
1439
- background-color: #efefef;
1440
- color: #000;
1441
- }
1442
- }
1443
- .first-name {
1444
- width: 40px;
1445
- height: 40px;
1446
- line-height: 40px;
1447
- text-align: center;
1448
- background: #5585f5;
1449
- color: #fff;
1450
- border-radius: 100%;
1451
- font-size: 15px;
1452
- }
1453
- .msg-avatar {
1454
- width: 40px;
1455
- height: 40px;
1456
- border-radius: 50%;
1457
-
1458
- object-fit: cover;
1459
- background: #fff;
1460
- }
1461
- /deep/ .ant-spin-spinning {
1462
- display: flex;
1463
- align-items: center;
1464
- }
1465
- .system-msg {
1466
- margin: 10px 0;
1467
- text-align: left;
1468
- line-height: 20px;
1469
- word-wrap: break-word;
1470
- display: flex;
1471
- justify-content: center;
1472
- .system-icon {
1473
- font-size: 18px;
1474
- color: #ffc200;
1475
- vertical-align: middle;
1476
- margin-right: 10px;
1477
- margin-top: 1px;
1478
- }
1479
- .system-btn {
1480
- color: @primary-color;
1481
- margin-left: 8px;
1482
- cursor: pointer;
1483
- text-decoration: underline;
1484
- }
1485
- }
1486
- }
1487
- /deep/ p {
1488
- margin-bottom: 0;
1489
- }
1490
- </style>
1
+ <template>
2
+ <div class="message-wrapper">
3
+ <chat-header :hideHeader="hideHeader" @openPortraitPanel="handleOpenPortraitPanel" v-if="isServer && enable"></chat-header>
4
+ <a-layout-content>
5
+ <slot></slot>
6
+ <div v-if="showBroadcast" class="broadcast-wrap">
7
+ <div v-if="broadcastCarousel" class="carousel-item" :class="{ 'has-link': assemblySetting.broadcast.PC.list[0].url }" @click="goToBroadcastLink(assemblySetting.broadcast.PC.list[0])">
8
+ <a-icon type="sound" />
9
+ <span class="ellips">
10
+ {{ assemblySetting.broadcast.PC.list[0].content }}
11
+ </span>
12
+ </div>
13
+ <template v-else>
14
+ <a-carousel dotPosition="right" autoplay :dots="false">
15
+ <div class="carousel-item" v-for="(item, i) in assemblySetting.broadcast.PC.list" :key="i" :class="{ 'has-link': item.url }" @click="goToBroadcastLink(item)">
16
+ <a-icon type="sound" />
17
+ <span class="ellips">{{ item.content }}</span>
18
+ </div>
19
+ </a-carousel>
20
+ </template>
21
+ </div>
22
+ <div v-if="!isHLWYY && sessionHistoryList.length > 1 && !(!onChating && currentTab === 'end') && !sessionEnd" style="position: absolute; top: 0; left: 20px; z-index: 1">
23
+ <a-radio-group :value="selectedSession" @change="handleSessionIdChange" buttonStyle="solid">
24
+ <a-radio-button :value="item.id + '-' + item.fromId" v-for="item in sessionHistoryList" :key="item.id + item.fromId">{{ item.fromName }}</a-radio-button>
25
+ </a-radio-group>
26
+ </div>
27
+ <div class="online-message" ref="message-wrapper" v-infinite-scroll="handleInfiniteOnLoad" @scroll="scrollEvent">
28
+ <template v-for="(item, index) in msgList">
29
+ <div :key="index" v-if="item">
30
+ <p v-if="isNeedShowTime(item, index)" class="message-time">
31
+ {{ formatDate(item.createdTime) }}
32
+ </p>
33
+ <div v-if="item.recallFlag" style="margin: 10px 0">
34
+ <template v-if="handleRecallFlag(item)">
35
+ {{ getRecallText(item, isServer) }}
36
+ <a style="text-decoration: underline" @click="reEditMessage(item)" v-if="[0, 4].includes(item.content && item.content.type) && onChating && isServer">,{{ i18nText('1.9.368') }}</a>
37
+ </template>
38
+ <template v-else>
39
+ {{ getRecallText(item) || `${item.fromName} ${i18nText('1.9.365')}` }}
40
+ </template>
41
+ </div>
42
+ <!-- 系统消息 -->
43
+ <div class="system-msg" v-else-if="isSystemMsg(item)">
44
+ <a-icon class="system-icon" theme="filled" type="exclamation-circle" />
45
+ <div>
46
+ <span v-html="getSystemMsgContent(item)"></span>
47
+ <span class="system-btn" v-for="(btn, index) in getTemplateButton(item)" @click="handleTemplateButton(btn)" :key="index">{{ btn.title }}</span>
48
+ </div>
49
+ </div>
50
+ <div
51
+ v-else-if="!item.recallFlag"
52
+ class="message-item"
53
+ :class="[getMessageItemClass(item), { 'read-status': enableReadRecord && item.readStatus }]"
54
+ :data-rid="item.readStatus === 'N' ? item.id : ''"
55
+ >
56
+ <img v-if="!isServer && item.fromPortrait" class="msg-avatar" :src="item.fromPortrait" />
57
+
58
+ <div v-else-if="!isServer && item.fromName" class="first-name">
59
+ {{ item.fromName && item.fromName.slice(0, 2) }}
60
+ </div>
61
+ <img v-else-if="!isServer" class="msg-avatar" :src="getAvatar(item)" width="40" height="40" />
62
+
63
+ <template v-else>
64
+ <img :src="customerStaffIcon" class="msg-avatar" width="40" height="40" v-if="item.fromId === 'SYSTEM'" />
65
+
66
+ <img src="../img/system_message.png" class="msg-avatar" width="40" height="40" v-else-if="item.content && item.content.source === 1" />
67
+
68
+ <img v-else-if="item.portrait" class="msg-avatar" :src="item.portrait" />
69
+
70
+ <div v-else class="first-name">
71
+ {{ item.fromName && item.fromName.slice(0, 2) }}
72
+ </div>
73
+ </template>
74
+ <div @contextmenu="handleContextmenu($event, item)" style="display: flex">
75
+ <a-spin v-if="item.sending">
76
+ <a-icon slot="indicator" type="loading" style="font-size: 24px" spin />
77
+ </a-spin>
78
+ <a-popconfirm @confirm="handleSendFail(item)" :okText="i18nText('1.1.1.1.3')" :cancelText="i18nText('2.7.1.14')">
79
+ <a-icon type="exclamation-circle" slot="icon" />
80
+ <div slot="title">
81
+ <span>确认重新发送该消息?</span>
82
+ </div>
83
+ <a-icon v-if="item.fail" theme="filled" type="exclamation-circle" style="font-size: 16px; display: flex; align-items: center; color: red" />
84
+ </a-popconfirm>
85
+ <div class="content-wrap" v-if="item.content.type === 0">
86
+ <span class="right-time">
87
+ {{ formatDate(item.createdTime || item.sendTime, true) }}
88
+ </span>
89
+ <div
90
+ :class="{
91
+ content: true,
92
+ 'has-dictionary': item.dictionaryValues && item.dictionaryValues.length,
93
+ 'has-intention': item.intentionList && item.intentionList.length
94
+ }"
95
+ >
96
+ <p v-html="getContentHtml(item)"></p>
97
+ <template v-if="item.content.attachments && item.content.attachments.length > 0">
98
+ <div v-for="(v, index) in item.content.attachments" :key="index" class="robot-item">
99
+ <a v-if="v.type == 1" :href="v.url" target="_blank"> {{ v.desc }}(超链) </a>
100
+ <a @click.prevent.stop="$emit('sendQuestion', v, item)" v-if="v.type == 2 || v.type == 3" href="javascript:void(0);">
101
+ {{ v.desc }}
102
+ </a>
103
+ </div>
104
+ </template>
105
+ <div v-if="item.intentionList && item.intentionList.length" class="intention-wrap">
106
+ <div v-for="(v, index) in item.intentionList" :key="index" @click.prevent.stop="$emit('sendIntention', v)">
107
+ <svg-icon :icon-class="v.icon || 'moren'" style="font-size: 26px"></svg-icon>
108
+ <span>{{ v.name }}</span>
109
+ </div>
110
+ </div>
111
+ </div>
112
+ <ul class="dictionary-ul" v-if="item.dictionaryValues && item.dictionaryValues.length">
113
+ <li v-for="(v, i) in item.dictionaryValues" :key="i" @click.prevent.stop="$emit('sendDictionary', v, item)">
114
+ {{ v }}
115
+ </li>
116
+ </ul>
117
+ <div class="client-btn" v-if="item.autoAccess && item.autoAccess === 'false'">
118
+ <svg-icon icon-class="kefu"></svg-icon>
119
+ <span @click="$emit('goCustomServe')">转人工客服</span>
120
+ </div>
121
+ </div>
122
+ <div v-else-if="item.content.type === 1" class="upload-image">
123
+ <MsgPicture style="max-width: 345px" :image="item.content.content" @click="show(item)" />
124
+ </div>
125
+ <div class="content-wrap" v-else-if="item.content.type === 2">
126
+ <span class="right-time">
127
+ {{ formatDate(item.createdTime || item.sendTime, true) }}
128
+ </span>
129
+ <div class="content">
130
+ <img style="width: 16px;margin-right:8px" src="../img/multi-video.png" />
131
+ {{ item.content.content }}&nbsp;
132
+ <span @click.prevent.stop="reCall(item, 'video')" class="re-call" v-if="item.content.isRefuse == 1">
133
+ {{ handleReCall(item) }}
134
+ </span>
135
+ </div>
136
+ </div>
137
+ <div class="content-wrap" v-else-if="item.content.type === 5">
138
+ <span class="right-time">
139
+ {{ formatDate(item.createdTime || item.sendTime, true) }}
140
+ </span>
141
+ <div class="content">
142
+ <img style="width: 16px;margin-right:8px" src="../img/multi-voice.png" />
143
+ {{ item.content.content }}&nbsp;
144
+ <span @click.prevent.stop="reCall(item, 'voice')" class="re-call" v-if="item.content.isRefuse == 1">
145
+ {{ handleReCall(item) }}
146
+ </span>
147
+ </div>
148
+ </div>
149
+ <div class="content-wrap" v-else-if="item.content.type === 6">
150
+ <span class="right-time">
151
+ {{ formatDate(item.createdTime || item.sendTime, true) }}
152
+ </span>
153
+ <div class="voice-bg content" @click.prevent.stop="playAudio(item)">
154
+ <img v-if="item.content.isPlay" src="../img/audio-play.gif" alt="" />
155
+ <img v-else src="../img/audio-new.png" alt="" />
156
+ <span class="voice-text" :style="AudioStyle(item)"> {{ item.content.duration || 1 }}" </span>
157
+ </div>
158
+ </div>
159
+ <template v-else-if="item.content.type === 4">
160
+ <msg-describe v-if="isDescribeMsg(item)" :item="item" />
161
+ <div v-else class="content-wrap">
162
+ <span class="right-time">
163
+ {{ formatDate(item.createdTime || item.sendTime, true) }}
164
+ </span>
165
+ <MsgPrescription :data="item" :isServer="isServer" :theme="isServer ? '' : 'customer'" @handleTemplateDetail="handleTemplateDetail" @handleTemplateButton="handleTemplateButton" />
166
+ </div>
167
+ </template>
168
+ </div>
169
+ <ReadStatus v-if="enableReadRecord" :status="item.readStatus" @click="openReadRecord($event, item)" />
170
+ </div>
171
+ </div>
172
+ </template>
173
+ <ul class="menu-main" ref="contextMenu" :style="contextMenuStyle" v-if="contextMenuShow">
174
+ <template v-for="(item, index) in rightClickSetting">
175
+ <li :key="index" class="menu-item" @click="() => handleRightClick(contextMenuItem, item)">
176
+ {{ item.rightClickName }}
177
+ </li>
178
+ </template>
179
+ </ul>
180
+ <div class="session-end" v-if="sessionEnd && !isServer">
181
+ <span class="end">{{ i18nText('1.9.370') }}</span>
182
+ </div>
183
+ <a-spin v-if="loading" class="loading-icon" />
184
+ </div>
185
+ <!-- 图片预览 -->
186
+ <viewer class="viewer" ref="viewer" :images="images" @inited="inited" :options="options" style="display: none">
187
+ <template slot-scope="scope">
188
+ <img v-for="src in scope.images" :src="src" :key="src" />
189
+ </template>
190
+ </viewer>
191
+ <!-- 卡片详情 -->
192
+ <a-modal
193
+ wrapClassName="chat-footer-modal-small"
194
+ :title="modalData.title || i18nText('1.1.1.10')"
195
+ :visible="modalShow"
196
+ :maskClosable="false"
197
+ :mask="true"
198
+ width="80%"
199
+ destroyOnClose
200
+ :okText="i18nText('1.2.1.11.6')"
201
+ :footer="null"
202
+ @cancel="handleClose"
203
+ >
204
+ <iframe id="chat-footer-modal" v-if="modalData.targetType === 'LINK_ADDRESS'" width="100%" height="100%" frameborder="0" :src="modalData.address" allow="camera;midi"></iframe>
205
+ </a-modal>
206
+ </a-layout-content>
207
+ <audio style="display: none" controls autoplay ref="audio" preload="auto" crossOrigin="anonymous" media-player="audioPlayer" :src="curAudioUrl" @ended="audioEnd"></audio>
208
+ <ReadRecord v-if="readRecordVisible" :style="readRecordStyle" :list="readRecordList" @close="closeReadRecord" />
209
+ </div>
210
+ </template>
211
+
212
+ <script>
213
+ import { Layout, Modal, Icon, Spin, Popconfirm, Carousel, Radio } from 'ant-design-vue';
214
+ import SvgIcon from '@/component/svg/index.vue';
215
+ import InfiniteScroll from '@/directive/scroll';
216
+ import { mapGetters, mapMutations, mapActions } from '../store/helper';
217
+ import fetch, { qs } from '@/utils/chatFetch';
218
+ import MsgDescribe from '../components/msg-describe';
219
+ import MsgPicture from '../components/msg-picture';
220
+ import MsgPrescription from '../components/msg-prescription';
221
+ import ReadStatus from '../components/read-status';
222
+ import ReadRecord from '../components/read-record';
223
+ import chatHeader from './chatHeader';
224
+ import viewerOptions from './mixins/viewerOptions';
225
+ import { getRecallText } from '../utils';
226
+ import vexutils from '@/utils/vexutils';
227
+ import ObserverScroll from '../utils/observer-scroll';
228
+
229
+ const customerStaffIcon = require('../img/customer_staff.png');
230
+ export default {
231
+ mixins: [viewerOptions],
232
+ inject: ['store', 'dispatchEvent', 'registerEvent', 'unregisterEvent', 'i18nText'],
233
+ props: {
234
+ hideHeader: {
235
+ type: Boolean,
236
+ default: false
237
+ },
238
+ robotAvatar: {
239
+ type: String,
240
+ default: ''
241
+ },
242
+ curChatType: {
243
+ type: String
244
+ },
245
+ isShowPortraitPanel: {
246
+ type: Boolean
247
+ },
248
+ activatedTime: {
249
+ type: Number
250
+ }
251
+ },
252
+ data() {
253
+ return {
254
+ loading: false,
255
+ // 卡片详情
256
+ modalShow: false,
257
+ modalData: {},
258
+ // 客户头像
259
+ // 右键操作
260
+ contextMenuStyle: {
261
+ position: 'fixed',
262
+ top: 0,
263
+ left: 0
264
+ },
265
+ contextMenuShow: false,
266
+ contextMenuItem: null,
267
+ // 查看历史消息滚动定位
268
+ scrollHeight: 0,
269
+ curAudioUrl: '',
270
+ lastAudioItem: null,
271
+ readRecordStyle: '',
272
+ readRecordVisible: false,
273
+ readRecordList: []
274
+ };
275
+ },
276
+ computed: {
277
+ ...mapGetters([
278
+ 'userInfo',
279
+ 'currentTab',
280
+ 'isAppendMsg',
281
+ 'message',
282
+ 'msgList',
283
+ 'sessionId',
284
+ 'serviceId',
285
+ 'scrollTo',
286
+ 'isServer',
287
+ 'clientId',
288
+ 'sessionEnd',
289
+ 'assemblyId',
290
+ 'footerMessage',
291
+ 'portraitPanelParams',
292
+ 'assemblySetting',
293
+ 'clientParams',
294
+ 'onChating',
295
+ 'classify',
296
+ 'collapsed',
297
+ 'barStatus',
298
+ 'sessionHistoryList',
299
+ 'enable',
300
+ 'appendList',
301
+ 'queueItem',
302
+ 'isRecorderVoice',
303
+ 'callTimer',
304
+ 'requestId'
305
+ ]),
306
+ // 互联网医院聊天组件
307
+ isHLWYY() {
308
+ return this.assemblyId === '1227514588661026816'
309
+ },
310
+ orgId() {
311
+ const getters = this.store.getters;
312
+ return this.isServer ? this.userInfo?.sysParams?.orgId : getters.accessParams.orgId || getters.queryParams.orgId;
313
+ },
314
+ userId() {
315
+ const getters = this.store.getters;
316
+ return this.isServer ? this.userInfo?.sysParams?.userId : getters.accessParams.userId || getters.chatUserId || getters.queryParams.userId;
317
+ },
318
+ enableReadRecord() {
319
+ return this.assemblySetting?.readStatus === 'Y';
320
+ },
321
+ customerStaffIcon() {
322
+ return this.robotAvatar || customerStaffIcon;
323
+ },
324
+ selectedSession() {
325
+ let sessionMap = this.sessionHistoryList.find(item => item.fromId === this.serviceId);
326
+ if (sessionMap) {
327
+ return sessionMap.id + '-' + sessionMap.fromId;
328
+ }
329
+ return '';
330
+ },
331
+ isListClassify() {
332
+ return this.assemblySetting.isListClassify === 'Y';
333
+ },
334
+ showPanel() {
335
+ return this.assemblySetting.portraitPanelSetting?.targetIs;
336
+ },
337
+ showBroadcast() {
338
+ let res = (!this.isServer && this.assemblySetting?.broadcast?.PC?.isChecked == 'Y' && this.assemblySetting?.broadcast?.PC?.list?.length > 0) || false;
339
+ return res;
340
+ },
341
+ broadcastCarousel() {
342
+ return (this.assemblySetting.broadcast && this.assemblySetting.broadcast.PC.list.length == 1) || false;
343
+ },
344
+ rightClickSetting() {
345
+ const { isServer } = this;
346
+ if (!isServer) return [];
347
+ const isSelf = this.contextMenuItem.fromId === this.serviceId;
348
+ const setting = this.assemblySetting?.rightClickSetting || [];
349
+ const doArr = ['resend', 'rollBack'];
350
+ return setting.filter(item => {
351
+ const { rightClickDo } = item;
352
+ return isSelf || doArr.indexOf(rightClickDo) < 0;
353
+ });
354
+ },
355
+ handleReCall() {
356
+ return function(item) {
357
+ if (item.content?.source === 1) {
358
+ return '[点击回拨]';
359
+ } else if (item.senderFlag == 1) {
360
+ return `[点击${this.isServer ? '回拨' : '重拨'}]`;
361
+ }
362
+ if (item.senderFlag == 2) {
363
+ return `[点击${this.isServer ? '重拨' : '回拨'}]`;
364
+ }
365
+ if (this.isServer) {
366
+ let isRight = item.fromId !== 'SYSTEM' && item.fromId && item.fromId === this.serviceId;
367
+ return `[点击${isRight ? '重拨' : '回拨'}]`;
368
+ } else {
369
+ let isRight = (item.fromId !== 'SYSTEM' && item.toId === 'SYSTEM') || item.fromId === this.clientId || (item.mode === 'bot' && !item.fromId);
370
+ return `[点击${isRight ? '重拨' : '回拨'}]`;
371
+ }
372
+ };
373
+ },
374
+ handleRecallFlag() {
375
+ return function(item) {
376
+ return this.serviceId == item.fromId;
377
+ };
378
+ },
379
+ AudioStyle() {
380
+ return function(item) {
381
+ let { duration = 1 } = item.content;
382
+ if (!duration) {
383
+ duration = 1;
384
+ }
385
+ let curWidth = (+duration / 60) * 400;
386
+ return {
387
+ width: curWidth < 13 ? 13 : curWidth + 'px'
388
+ };
389
+ };
390
+ },
391
+ showTimeDuration() {
392
+ let { variableSetting = null } = this.assemblySetting || {};
393
+ if (!variableSetting) return 3;
394
+ let matchItem = variableSetting.find(v => v.name === 'CHAT_SEPARATION_TIME');
395
+ if (!matchItem) return 3;
396
+ if (!+matchItem.value) return 3;
397
+ return +matchItem.value;
398
+ },
399
+ images() {
400
+ return this.msgList.filter(v => !v.recallFlag && v.content.type == 1)?.map(v => v.content?.content);
401
+ }
402
+ },
403
+ created() {
404
+ this.attachEvent();
405
+ this.getUserInformation();
406
+ },
407
+ methods: {
408
+ ...mapMutations([
409
+ 'setIsAppendMsg',
410
+ 'setMsgList',
411
+ 'setScrollTo',
412
+ 'setFooterMessage',
413
+ 'setIsScrollListChange',
414
+ 'setListChangeItem',
415
+ 'setSessionId',
416
+ 'setOnChating',
417
+ 'setServiceId',
418
+ 'setShowAudio',
419
+ 'setVideoMode',
420
+ 'setVideoMembers',
421
+ 'setIsRecorderVoice',
422
+ 'setDoctorInfo',
423
+ 'setCallTimer'
424
+ ]),
425
+ ...mapActions({
426
+ getEarlierMsg: 'setMsgList',
427
+ handleBotChat: 'handleBotChat',
428
+ sendMessage: 'sendMessage'
429
+ }),
430
+ // 获取医生信息
431
+ getUserInformation() {
432
+ const url = '/user/getUserInformation';
433
+ fetch.get(url).then(({ data }) => {
434
+ if (data.result === 'SUCCESS') {
435
+ this.setDoctorInfo(data?.map?.sysUserInformation)
436
+ console.log('data?.map?.sysUserInformation', data?.map?.sysUserInformation);
437
+ }
438
+ });
439
+ },
440
+ getMessageItemClass(item) {
441
+ // content.source === 1, 代表为系统消息, 系统统一左侧展示
442
+ if (item.content?.source === 1) {
443
+ return {
444
+ left: true
445
+ };
446
+ }
447
+
448
+ // 服务端 senderFlag 1 左 2 右
449
+ // 客户端 senderFlag 1 右 2 左
450
+ if (item.content?.source === 1) {
451
+ return {
452
+ left: true
453
+ };
454
+ } else if (item.senderFlag == 1) {
455
+ if (this.isServer) {
456
+ return {
457
+ left: true
458
+ };
459
+ } else {
460
+ return {
461
+ right: true
462
+ };
463
+ }
464
+ }
465
+ if (item.senderFlag == 2) {
466
+ if (this.isServer) {
467
+ return {
468
+ right: true
469
+ };
470
+ } else {
471
+ return {
472
+ left: true
473
+ };
474
+ }
475
+ }
476
+ if (this.isServer) {
477
+ return {
478
+ right: item.fromId !== 'SYSTEM' && item.fromId && item.fromId === this.serviceId,
479
+ left: item.fromId === 'SYSTEM' || item.fromId === this.clientId || item.content?.source === 1 || !item.fromId
480
+ };
481
+ } else {
482
+ let isRight, isLeft;
483
+ if (this.curChatType == 'robot') {
484
+ isRight = item.fromId !== 'SYSTEM';
485
+ isLeft = item.fromId === 'SYSTEM';
486
+ } else {
487
+ isRight = (item.fromId !== 'SYSTEM' && item.toId === 'SYSTEM') || item.fromId === this.clientId || (item.mode === 'bot' && !item.fromId);
488
+ isLeft = item.fromId === 'SYSTEM' || item.fromId !== this.clientId || item.content?.source === 1;
489
+ }
490
+ return {
491
+ right: isRight,
492
+ left: isLeft
493
+ };
494
+ }
495
+ },
496
+ /* 向上滚动刷新 */
497
+ handleInfiniteOnLoad() {
498
+ if (this.curChatType === 'robot') return;
499
+ this.loading = true;
500
+ this.getEarlierMsg().then(msg => {
501
+ if (msg) {
502
+ this.$message.info(msg);
503
+ }
504
+ this.loading = false;
505
+ });
506
+ },
507
+ show(item) {
508
+ const src = item.content.content;
509
+ // this.images = vexutils.imgs2imgArr(src);
510
+ this.$viewer.view(this.images.findIndex(v => v === src));
511
+ this.$viewer.show();
512
+ this.readSingleMsg(item);
513
+ },
514
+ /* 滚动条定位 */
515
+ handleScrollTo(scrollTo) {
516
+ let target = this.$refs['message-wrapper'];
517
+ this.$nextTick().then(() => {
518
+ if (!target) return;
519
+ let top = target.scrollHeight;
520
+ if (scrollTo === 'current') {
521
+ top = target.scrollHeight - this.scrollHeight;
522
+ }
523
+ if (typeof scrollTo === 'number') {
524
+ top = scrollTo;
525
+ }
526
+ target.scrollTo({
527
+ left: 0,
528
+ top,
529
+ behavior: 'instant'
530
+ });
531
+ this.scrollHeight = 0;
532
+ });
533
+ this.setScrollTo('');
534
+ },
535
+
536
+ /**
537
+ * 是否需要显示时间
538
+ */
539
+ isNeedShowTime(item, index) {
540
+ const curMsgTime = item.createdTime || new Date().getTime();
541
+ let duration = 60 * 1000 * this.showTimeDuration;
542
+ let lastMsgTime;
543
+ if (index == 0) {
544
+ lastMsgTime = 0;
545
+ } else {
546
+ lastMsgTime = this.msgList[index - 1].createdTime;
547
+ }
548
+ if (Number(curMsgTime) - Number(lastMsgTime) > duration) {
549
+ return true;
550
+ } else {
551
+ return false;
552
+ }
553
+ },
554
+ formatDate(date) {
555
+ return vexutils.formatDate(date, true);
556
+ },
557
+ getAvatar(item) {
558
+ if (item.content && item.content.source === 1) {
559
+ return require('../img/system_message.png');
560
+ }
561
+ if (item.mode === 'bot' && !item.fromId) {
562
+ return require('../img/default.png');
563
+ }
564
+ if (this.isServer) {
565
+ if (item.fromId === this.clientId || !item.fromId) {
566
+ return require('../img/default.png');
567
+ } else {
568
+ return this.customerStaffIcon;
569
+ }
570
+ } else {
571
+ if (this.curChatType == 'robot') {
572
+ if (item.fromId === 'SYSTEM') {
573
+ return this.customerStaffIcon;
574
+ } else {
575
+ return require('../img/default.png');
576
+ }
577
+ } else {
578
+ if (item.fromId !== this.clientId) {
579
+ return this.customerStaffIcon;
580
+ } else {
581
+ return require('../img/default.png');
582
+ }
583
+ }
584
+ }
585
+ },
586
+ getContent(item) {
587
+ let content;
588
+ if (vexutils.isJSON(item.content.content)) {
589
+ content = JSON.parse(item.content.content);
590
+ } else {
591
+ content = item.content.content;
592
+ }
593
+ return content;
594
+ },
595
+ getContentHtml(item) {
596
+ return !~item.content.content.indexOf('emoji') ? this.$xss(item.content.content) : item.content.content;
597
+ },
598
+ getTemplateTitle(item) {
599
+ return this.getContent(item)?.title;
600
+ },
601
+ showArrow(item) {
602
+ let setting = this.getContent(item)?.customerSetting || {};
603
+ if (this.isServer) {
604
+ setting = this.getContent(item)?.serverSetting || {};
605
+ }
606
+ let { address } = setting;
607
+ if (!address) return false;
608
+ return true;
609
+ },
610
+ getTemplateTitleIcon(item) {
611
+ return this.getContent(item)?.icon;
612
+ },
613
+ getTemplateContent(item) {
614
+ let content = this.getContent(item);
615
+ let array = [];
616
+ try {
617
+ array = content.content.split('##');
618
+ } catch {
619
+ console.log(content, 'error');
620
+ }
621
+ array = array.map(item => `<div>${item}</div>`);
622
+ return array.join('');
623
+ },
624
+ getSystemMsgContent(item) {
625
+ let content = this.getContent(item);
626
+ let array = [];
627
+ try {
628
+ array = content.content.split('##');
629
+ } catch {
630
+ console.log(content, 'error');
631
+ }
632
+ return array.join('<br>');
633
+ },
634
+ getTemplateButton(item) {
635
+ let content = this.getContent(item);
636
+ const setting = this.isServer ? content.serverSetting : content.customerSetting;
637
+ return setting?.toolbar_button || [];
638
+ },
639
+ // 处理模板按钮,详情等跳转
640
+ handleTemplate(data, dispatch = true) {
641
+ let { targetType, address, params = [], openMode } = data;
642
+ dispatch &&
643
+ this.dispatchEvent('click_templateCard', {
644
+ ...data
645
+ });
646
+ if (!address) return;
647
+ if (targetType === 'LINK_ADDRESS') {
648
+ let urlParams = [];
649
+ params.forEach(({ p_name, p_value }) => {
650
+ urlParams.push(`${p_name}=${p_value}`);
651
+ });
652
+ if (address.includes('?')) {
653
+ address += `&${urlParams.join('&')}`;
654
+ } else {
655
+ address += `?${urlParams.join('&')}`;
656
+ }
657
+ if (openMode === 'WINDOW') {
658
+ window.open(address);
659
+ return;
660
+ }
661
+ if (openMode === 'EJECT') {
662
+ // 打开云his页面需要从cookie内获取最新的jsessionids
663
+ if (address.includes('jsessionidsFlag=1')) {
664
+ let jsessionids = vexutils.cookie.get('jsessionids')
665
+ address = address + `&jsessionids=${jsessionids}`
666
+ }
667
+ this.modalShow = true;
668
+ this.modalData = Object.assign({}, data, {
669
+ address
670
+ });
671
+ this.$nextTick().then(() => {
672
+ window.addEventListener('message', this.iframeEvent);
673
+ });
674
+ return;
675
+ }
676
+ }
677
+ },
678
+ iframeEvent(event) {
679
+ const method = event?.data?.method;
680
+ this[method]?.(event);
681
+ },
682
+ // 发送
683
+ handleParentMessageSend({ data, source, origin }) {
684
+ let params = {
685
+ assemblyId: this.assemblyId,
686
+ sessionId: this.sessionId,
687
+ orgId: this.orgId
688
+ };
689
+ params = Object.assign(params, data.data);
690
+ return fetch.post('/chat/access/sendToolBarData', qs.stringify(params)).then(({ data }) => {
691
+ if (data.result === 'SUCCESS') {
692
+ source.postMessage({ status: 0, resultMsg: data.resultMsg }, origin);
693
+ } else {
694
+ source.postMessage({ status: 1, resultMsg: data.resultMsg }, origin);
695
+ }
696
+ });
697
+ },
698
+ // 关闭
699
+ handleClose() {
700
+ this.modalShow = false;
701
+ window.removeEventListener('message', this.iframeEvent);
702
+ },
703
+ handleTemplateDetail(item) {
704
+ let content = this.getContent(item) || {};
705
+ let setting = content.customerSetting || {};
706
+ if (this.isServer) {
707
+ setting = content.serverSetting || {};
708
+ }
709
+ this.readSingleMsg(item);
710
+ this.handleTemplate(setting);
711
+ },
712
+ handleTemplateButton(btn) {
713
+ this.handleTemplate(btn);
714
+ },
715
+ // 消息撤回
716
+ handleRecall(item) {
717
+ let params = {
718
+ assemblyId: this.assemblyId,
719
+ sessionId: this.sessionId,
720
+ messageId: item.id,
721
+ messageType: item.content?.type,
722
+ userId: this.userInfo?.sysParams?.userId || ''
723
+ };
724
+
725
+ fetch.post('/chat/service/recallMessage', qs.stringify(params)).then(({ data }) => {
726
+ if (data.result === 'SUCCESS') {
727
+ this.dispatchEvent('msg_recallSuccess', {
728
+ message: { ...item },
729
+ ...params
730
+ });
731
+ this.$set(item, 'recallFlag', 1);
732
+ // 撤回成功提示
733
+ let title = this.getContent(item)?.title;
734
+ let text = title ? `已撤回[${title}]` : '已撤回';
735
+ this.$message.success(text);
736
+
737
+ if (!this.isLastMessage(item.id)) return;
738
+ this.setIsScrollListChange(true);
739
+ this.setListChangeItem({
740
+ opt: 1,
741
+ from: this.clientId,
742
+ sessionId: this.sessionId,
743
+ msgCount: 0,
744
+ lastContentRecallFlag: 1,
745
+ lastContent: vexutils.clone(item.content, true),
746
+ lastTime: new Date().getTime()
747
+ });
748
+ } else {
749
+ this.$message.error(data.resultMsg);
750
+ }
751
+ });
752
+ },
753
+ handleJumpUrl(contextMenuItem, item) {
754
+ const title = item.rightClickName;
755
+ const url = item.rightClickUrl.replace(/{(.*?)}/g, (match, p) => {
756
+ return eval(p);
757
+ });
758
+ this.dispatchEvent('open_jumpUrl', {
759
+ title: title,
760
+ url: url
761
+ });
762
+ },
763
+ // 右键菜单功能
764
+ handleRightClick(contextMenuItem, item) {
765
+ if (item.rightClickDo === 'rollBack') {
766
+ this.handleRecall(contextMenuItem);
767
+ } else if (item.rightClickDo === 'jumpUrl') {
768
+ this.handleJumpUrl(contextMenuItem, item);
769
+ } else if (item.rightClickDo === 'resend') {
770
+ this.resendMessage(contextMenuItem);
771
+ }
772
+ },
773
+ resendMessage(item) {
774
+ fetch({
775
+ method: 'POST',
776
+ url: '/chat/service/resendMessage',
777
+ data: qs.stringify({
778
+ assemblyId: this.assemblyId,
779
+ sessionId: this.sessionId,
780
+ messageId: item.id
781
+ })
782
+ });
783
+ },
784
+ // 重新编辑
785
+ reEditMessage(item) {
786
+ if (item.content?.type === 4) {
787
+ let content = this.getContent(item);
788
+ this.handleTemplate(content.serverSetting.reEditSetting, false);
789
+ return;
790
+ }
791
+ this.setFooterMessage(this.footerMessage + item.content.content.slice(5, -6));
792
+ },
793
+ observeMsg() {
794
+ if (this.enableReadRecord) {
795
+ if (!this._observer) {
796
+ this._observer = new ObserverScroll(this.$refs['message-wrapper'], this.readMsg);
797
+ this.$on('hook:beforeDestroy', () => {
798
+ this._observer.remove();
799
+ this._observer = null;
800
+ });
801
+ }
802
+ this._observer.disconnect();
803
+ if (this.msgList.length) {
804
+ this.$nextTick(() => {
805
+ const el = this.$refs['message-wrapper'];
806
+ const els = el?.querySelectorAll('.message-item.left');
807
+ els && this._observer.observe(Array.from(els).filter(item => item.getAttribute('data-rid')));
808
+ });
809
+ }
810
+ }
811
+ },
812
+ readMsg(ids, enableVaid = true, notify = true, post = true) {
813
+ const msgList = this.msgList;
814
+ const idKeys = msgList.reduce((obj, item, i) => {
815
+ const id = item?.id;
816
+ if (id) obj[id] = i;
817
+ return obj;
818
+ }, {});
819
+ const data = [];
820
+ const validFn = (item, type) => {
821
+ /**
822
+ * 针对文本、表情消息类型,只要在聊天区域内,视为已读,
823
+ * 图片、语音类型消息,需要点击查看后,才视为已读,
824
+ * 模板 类型消息,如果没有详情链接,只要在聊天区域内,视为已读,若有详情链接,需要点击查看后,才视为已读
825
+ */
826
+ const jumpType = `,6,1,`;
827
+ if (jumpType.includes(type)) return false;
828
+ if (type == 4 && !this.isDescribeMsg(item)) {
829
+ let content = this.getContent(item) || {};
830
+ let setting = (this.isServer ? content.serverSetting : content.customerSetting) || {};
831
+ return !setting?.address;
832
+ }
833
+ return true;
834
+ };
835
+ const value = ids.filter(id => {
836
+ const i = idKeys[id];
837
+ let item = msgList[i];
838
+ if (item) {
839
+ const type = item.content.type;
840
+ if (!enableVaid || validFn(item, type)) {
841
+ item.readStatus = 'Y';
842
+ data.push({ id, type });
843
+ return true;
844
+ }
845
+ return false;
846
+ }
847
+ });
848
+ if (!value.length) return;
849
+ notify &&
850
+ this.dispatchEvent('msg_readed', {
851
+ data,
852
+ assemblyId: this.assemblyId,
853
+ sessionId: this.sessionId
854
+ });
855
+ post && this.postReadMessage({ ids: value.join(',') });
856
+ },
857
+ postReadMessage(data) {
858
+ fetch.post(
859
+ `/chat/${this.isServer ? 'service' : 'access'}/readMessage`,
860
+ qs.stringify({
861
+ assemblyId: this.assemblyId,
862
+ orgId: this.orgId,
863
+ userId: this.userId,
864
+ sessionId: this.sessionId,
865
+ ...data
866
+ })
867
+ );
868
+ },
869
+ readSingleMsg(item) {
870
+ if (this.enableReadRecord && item.readStatus === 'N') {
871
+ const cls = this.getMessageItemClass(item);
872
+ if (!cls?.left) return;
873
+ this.readMsg([item.id], false);
874
+ }
875
+ },
876
+ closeReadRecord() {
877
+ this.readRecordVisible = false;
878
+ this.readRecordList = [];
879
+ },
880
+ openReadRecord(event, item) {
881
+ const readStatus = item.readStatus;
882
+ if (readStatus) {
883
+ const cls = this.getMessageItemClass(item);
884
+ if (!cls.right) return;
885
+ const rect = event.target.getBoundingClientRect();
886
+ const w = 440;
887
+ const h = 340;
888
+ const top = Math.max(0, Math.min(document.body.clientHeight - h, rect.top - h / 2));
889
+ const left = Math.max(0, Math.min(document.body.clientWidth - w, rect.left - w));
890
+ fetch({
891
+ url: `/chat/${this.isServer ? 'service' : 'access'}/getMessageMember`,
892
+ method: 'get',
893
+ params: {
894
+ assemblyId: this.assemblyId,
895
+ messageId: item.id,
896
+ orgId: this.isServer ? undefined : this.orgId
897
+ }
898
+ }).then(res => {
899
+ const list = res.data?.list || [];
900
+ this.readRecordStyle = `top:${top}px;left:${left}px;position:fixed;z-index:10`;
901
+ this.readRecordVisible = list.length > 0;
902
+ this.readRecordList = Object.freeze(list);
903
+ if (item.readStatus && list.length) {
904
+ const unread = list.filter(item => item.readStatus == 'N').length;
905
+ const single = list.length < 2;
906
+ const status = unread > 0 ? (single ? 'N' : `${unread}`) : single ? 'Y' : 'ALL';
907
+ // 检查是否更新人数
908
+ if (item.readStatus != status) {
909
+ item.readStatus = status;
910
+ }
911
+ }
912
+ });
913
+ }
914
+ },
915
+ isLastMessage(id) {
916
+ return this.msgList.findIndex(msg => msg.id === id) === this.msgList.length - 1;
917
+ },
918
+ isSystemMsg(item) {
919
+ return item.content.type == 4 && this.getContent(item).styleId == '2';
920
+ },
921
+ isDescribeMsg(item) {
922
+ return this.getContent(item).styleId == '3';
923
+ },
924
+ async handleOpenPortraitPanel(item) {
925
+ if (!this.showPanel || item.fromId === this.serviceId || !this.isServer || !this.isShowPortraitPanel) return;
926
+ const { portraitPanelParams, clientParams } = this;
927
+ const panelSetting = this.assemblySetting?.portraitPanelSetting;
928
+ this.dispatchEvent('open_portraitPanel', {
929
+ panelSetting,
930
+ clientParams,
931
+ panelParams: portraitPanelParams,
932
+ targetId: panelSetting?.targetId,
933
+ targetType: panelSetting?.targetType
934
+ });
935
+ },
936
+ handleContextmenu(e, item) {
937
+ // 系统消息,客户消息,待接入消息,历史消息不允许撤回
938
+ // 模板-描述消息禁用右键
939
+ if (item.content?.source === 1 || !this.sessionId || (this.sessionId && !this.onChating) || (item.content.type === 4 && this.isDescribeMsg(item))) return;
940
+ this.contextMenuItem = item;
941
+ if (this.rightClickSetting.length < 1) return;
942
+ e.preventDefault();
943
+ let left = e.pageX + 'px';
944
+ let top = e.pageY + 'px';
945
+ this.contextMenuStyle = Object.assign(this.contextMenuStyle, {
946
+ top,
947
+ left
948
+ });
949
+ this.contextMenuShow = true;
950
+ },
951
+ scrollEvent(e) {
952
+ console.log('e.target.scrollTop->', e.target.scrollTop);
953
+ this._scrollTop = e.target.scrollTop;
954
+ },
955
+ /* 事件绑定 */
956
+ attachEvent() {
957
+ this.scrollEvent = vexutils.debounce(this.scrollEvent, 800);
958
+ const handleClick = () => {
959
+ this.contextMenuShow = false;
960
+ };
961
+ const handleRasize = () => {
962
+ if (this.readRecordVisible) {
963
+ let style = this.readRecordStyle;
964
+ const top = style.match(topReg)?.[1];
965
+ const topReg = /top:([\d.]+)/;
966
+ const maxH = document.body.clientHeight - 360;
967
+ if (top && maxH < top) {
968
+ style = style.replace(topReg, `top:${maxH}`);
969
+ }
970
+ const leftReg = /left:([\d.]+)/;
971
+ const left = style.match(leftReg)?.[1];
972
+ const maxW = document.body.clientWidth - 440;
973
+ if (left && maxW < left) {
974
+ style = style.replace(leftReg, `left:${maxW}`);
975
+ }
976
+ this.readRecordStyle = style;
977
+ }
978
+ };
979
+ window.addEventListener('resize', handleRasize);
980
+ document.addEventListener('click', handleClick);
981
+ this.registerEvent('openPortraitPanel', this.handleOpenPortraitPanel);
982
+ const readMessage = data => {
983
+ this.readMsg(data.ids.split(','), false, false, false);
984
+ this.postReadMessage(data);
985
+ };
986
+ this.registerEvent('readMessage', readMessage);
987
+ this.$on('hook:beforeDestroy', () => {
988
+ window.removeEventListener('resize', handleRasize);
989
+ document.removeEventListener('click', handleClick);
990
+ this.unregisterEvent('openPortraitPanel', this.handleOpenPortraitPanel);
991
+ this.unregisterEvent('readMessage', readMessage);
992
+ });
993
+ },
994
+ goToBroadcastLink(item) {
995
+ if (!item.url) return;
996
+ window.open(item.url);
997
+ },
998
+ /**
999
+ * 客服切换
1000
+ */
1001
+ handleSessionIdChange(e) {
1002
+ let [sessionId, serviceId] = e.target.value.split('-');
1003
+ this.setSessionId(sessionId);
1004
+ this.setServiceId(serviceId);
1005
+ let isOnChating = this.sessionHistoryList.find(item => item.fromId === serviceId).onChating;
1006
+ this.setOnChating(!!isOnChating);
1007
+ this.setMsgList([]);
1008
+ this.getEarlierMsg();
1009
+ },
1010
+ handleSendFail(item) {
1011
+ this.sendMessage(Object.assign(item.content, { failItem: item }));
1012
+ },
1013
+ reCall(type) {
1014
+ this.setShowAudio(true);
1015
+ this.setVideoMode(type === 'voice' ? 1 : 2);
1016
+ this.$nextTick(() => {
1017
+ this.setVideoMembers([]);
1018
+ });
1019
+ },
1020
+ playAudio(item) {
1021
+ let { content = '' } = item?.content || {};
1022
+ if (!content) return;
1023
+ if (this.lastAudioItem && (item.id !== this.lastAudioItem.id || item.createdTime !== this.lastAudioItem.createdTime)) {
1024
+ this.$set(this.lastAudioItem.content, 'isPlay', false);
1025
+ }
1026
+
1027
+ this.readSingleMsg(item);
1028
+ this.lastAudioItem = item;
1029
+ this.$set(item.content, 'isPlay', !item.content.isPlay);
1030
+ this.$nextTick(() => {
1031
+ this.curAudioUrl = content;
1032
+ let audioRef = this.$refs['audio'];
1033
+ if (!audioRef) return;
1034
+ audioRef.load();
1035
+ setTimeout(() => {
1036
+ if (item.content.isPlay) {
1037
+ let playPromise = audioRef.play();
1038
+ if (playPromise !== undefined) {
1039
+ playPromise.then(() => {
1040
+ audioRef.play();
1041
+ });
1042
+ }
1043
+ } else {
1044
+ audioRef.pause();
1045
+ this.curAudioUrl = '';
1046
+ }
1047
+ });
1048
+ });
1049
+ },
1050
+ audioEnd() {
1051
+ if (!this.lastAudioItem) return;
1052
+ this.$set(this.lastAudioItem.content, 'isPlay', false);
1053
+ },
1054
+ getRecallText(item, flag = false) {
1055
+ return getRecallText(this, item, flag);
1056
+ }
1057
+ },
1058
+ directives: {
1059
+ InfiniteScroll
1060
+ },
1061
+ components: {
1062
+ AModal: Modal,
1063
+ [Layout.Content.name]: Layout.Content,
1064
+ [Spin.name]: Spin,
1065
+ [Icon.name]: Icon,
1066
+ [Popconfirm.name]: Popconfirm,
1067
+ [Radio.Button.name]: Radio.Button,
1068
+ [Radio.Group.name]: Radio.Group,
1069
+ [Carousel.name]: Carousel,
1070
+ SvgIcon,
1071
+ MsgPrescription,
1072
+ ReadStatus,
1073
+ ReadRecord,
1074
+ MsgPicture,
1075
+ MsgDescribe,
1076
+ chatHeader
1077
+ },
1078
+ watch: {
1079
+ msgList() {
1080
+ this.observeMsg();
1081
+ },
1082
+ isAppendMsg: {
1083
+ immediate: true,
1084
+ handler(isAppendMsg) {
1085
+ if (isAppendMsg) {
1086
+ let newArr = [];
1087
+ if (vexutils.isPlainObject(this.message)) {
1088
+ newArr = [...this.msgList, this.message];
1089
+ } else if (vexutils.isArray(this.message)) {
1090
+ newArr = [...this.msgList, ...this.message];
1091
+ }
1092
+ this.setMsgList(newArr);
1093
+ this.setIsAppendMsg(false);
1094
+ this.setScrollTo('bottom');
1095
+ }
1096
+ }
1097
+ },
1098
+ scrollTo: {
1099
+ immediate: true,
1100
+ handler(scrollTo) {
1101
+ if (scrollTo) {
1102
+ this.handleScrollTo(scrollTo);
1103
+ }
1104
+ }
1105
+ },
1106
+ activatedTime: {
1107
+ immediate: true,
1108
+ handler(value) {
1109
+ this.handleScrollTo(this._scrollTop);
1110
+ }
1111
+ },
1112
+ /**
1113
+ * 图片懒加载处理
1114
+ */
1115
+ appendList: {
1116
+ handler() {
1117
+ if (!this.scrollHeight) {
1118
+ let target = this.$refs['message-wrapper'];
1119
+ this.scrollHeight = target.scrollHeight;
1120
+ }
1121
+ if (this.appendList.length === this.msgList.length || this.appendList.length === 1) {
1122
+ this.setScrollTo('bottom');
1123
+ } else {
1124
+ this.setScrollTo('current');
1125
+ }
1126
+ }
1127
+ },
1128
+ isRecorderVoice: {
1129
+ immediate: true,
1130
+ handler(val) {
1131
+ if (!val) return;
1132
+ this.curAudioUrl = '';
1133
+ this.setIsRecorderVoice(false);
1134
+ if (!this.lastAudioItem) return;
1135
+ this.$set(this.lastAudioItem.content, 'isPlay', false);
1136
+ }
1137
+ }
1138
+ }
1139
+ };
1140
+ </script>
1141
+
1142
+ <style lang="less" scoped>
1143
+ @import '../style/emoji.css';
1144
+ @import '../style/message.mixin.less';
1145
+ .message-wrapper {
1146
+ .createMessageStyle(#ebebeb, @primary-color);
1147
+ flex: 1;
1148
+ height: 0;
1149
+ // position: relative;
1150
+ display: flex;
1151
+ flex-direction: column;
1152
+ overflow-x: hidden;
1153
+ .broadcast-wrap {
1154
+ height: 30px;
1155
+ text-align: center;
1156
+ font-size: 14px;
1157
+ background-color: #fffbe8;
1158
+ /deep/ .carousel-item {
1159
+ display: inline-flex !important;
1160
+ justify-content: center;
1161
+ align-items: center;
1162
+ width: 100%;
1163
+ height: 30px;
1164
+ text-align: center;
1165
+ &.has-link {
1166
+ color: #ed6a0c;
1167
+ cursor: pointer;
1168
+ }
1169
+ span {
1170
+ display: inline-block;
1171
+ max-width: calc(100% - 30px);
1172
+ margin-left: 4px;
1173
+ }
1174
+ }
1175
+ }
1176
+
1177
+ .loading-icon {
1178
+ position: absolute;
1179
+ top: 0;
1180
+ left: 50%;
1181
+ }
1182
+ }
1183
+ /deep/ .ant-layout-content {
1184
+ flex: 1;
1185
+ display: flex;
1186
+ flex-direction: column;
1187
+ position: relative;
1188
+ }
1189
+ .online-message {
1190
+ flex: 1;
1191
+ height: 100%;
1192
+ // height: 608px;
1193
+ overflow-y: auto;
1194
+ overflow-x: hidden;
1195
+ padding: 10px 22px;
1196
+ // margin: 10px 0;
1197
+ text-align: center;
1198
+ &::-webkit-scrollbar {
1199
+ width: 5px;
1200
+ }
1201
+ .message-time {
1202
+ display: inline-block;
1203
+ color: #666;
1204
+ font-size: 14px;
1205
+ border-radius: 4px;
1206
+ padding: 4px;
1207
+ margin: 0 auto;
1208
+ margin-bottom: 16px;
1209
+ }
1210
+ > div {
1211
+ margin-bottom: 16px;
1212
+ }
1213
+ .message-item {
1214
+ display: flex;
1215
+ /deep/ .chat-read-status-text {
1216
+ display: none;
1217
+ }
1218
+ &.right.read-status {
1219
+ padding-bottom: 24px;
1220
+ position: relative;
1221
+ /deep/ .chat-read-status-text {
1222
+ display: block;
1223
+ position: absolute;
1224
+ right: 50px;
1225
+ bottom: 0;
1226
+ }
1227
+ }
1228
+ .right-time {
1229
+ display: none;
1230
+ position: absolute;
1231
+ top: -16px;
1232
+ left: 10px;
1233
+ font-size: 12px;
1234
+ color: #b0b1b1;
1235
+ white-space: nowrap;
1236
+ }
1237
+
1238
+ .content-wrap {
1239
+ padding: 0 10px;
1240
+ text-align: left;
1241
+ position: relative;
1242
+ &:hover {
1243
+ .right-time {
1244
+ display: block;
1245
+ }
1246
+ }
1247
+
1248
+ .robot-item {
1249
+ text-align: left;
1250
+ a {
1251
+ text-decoration: underline;
1252
+ }
1253
+ }
1254
+ .intention-wrap {
1255
+ display: flex;
1256
+ flex-wrap: wrap;
1257
+ padding: 8px 0 4px 0;
1258
+ background-color: #fff;
1259
+ border-radius: 0 0 8px 8px;
1260
+ > div {
1261
+ display: inline-flex;
1262
+ flex-direction: column;
1263
+ align-items: center;
1264
+ width: 60px;
1265
+ height: 54px;
1266
+ cursor: pointer;
1267
+
1268
+ &:hover {
1269
+ color: #5585f5;
1270
+ span {
1271
+ color: #5585f5;
1272
+ }
1273
+ }
1274
+ }
1275
+ span {
1276
+ margin-top: 8px;
1277
+ max-width: 90%;
1278
+ text-overflow: ellipsis;
1279
+ overflow: hidden;
1280
+ white-space: nowrap;
1281
+ font-size: 12px;
1282
+ color: #6a6a6a;
1283
+ }
1284
+ }
1285
+ .dictionary-ul {
1286
+ padding-top: 8px;
1287
+ display: flex;
1288
+ flex-wrap: wrap;
1289
+ li {
1290
+ padding: 4px 16px;
1291
+ margin-right: 16px;
1292
+ font-size: 14px;
1293
+ color: #5585f5;
1294
+ background: #efefef;
1295
+ border-radius: 16px;
1296
+ cursor: pointer;
1297
+ }
1298
+ }
1299
+ .client-btn {
1300
+ display: inline-block;
1301
+ padding: 4px 8px;
1302
+ margin-top: 8px;
1303
+ color: #fff;
1304
+ font-size: 12px;
1305
+ border-radius: 20px;
1306
+ background: #5585f5;
1307
+ cursor: pointer;
1308
+ }
1309
+ .re-call {
1310
+ cursor: pointer;
1311
+ color: #5585f5;
1312
+ }
1313
+ .voice-bg {
1314
+ display: flex;
1315
+ align-items: center;
1316
+ cursor: pointer;
1317
+ img {
1318
+ mix-blend-mode: difference;
1319
+ display: inline-block;
1320
+ width: 14px;
1321
+ }
1322
+ .voice-text {
1323
+ display: inline-block;
1324
+ }
1325
+ span {
1326
+ margin-left: 8px;
1327
+ }
1328
+ }
1329
+ }
1330
+ .has-dictionary {
1331
+ padding: 0;
1332
+ text-align: left;
1333
+ background: unset;
1334
+ }
1335
+ .content {
1336
+ padding: 10px;
1337
+ background: #ebebeb;
1338
+ border-radius: 0px 8px 8px 8px;
1339
+ max-width: 500px;
1340
+ margin: 0px;
1341
+ line-height: 20px;
1342
+ font-size: 14px;
1343
+ color: #000;
1344
+ text-align: left;
1345
+ word-break: break-word;
1346
+ /deep/ p {
1347
+ margin: 0;
1348
+ }
1349
+ }
1350
+ .upload-image {
1351
+ margin-left: 8px;
1352
+ }
1353
+ &.right {
1354
+ flex-direction: row-reverse;
1355
+ .right-time {
1356
+ left: unset;
1357
+ right: 10px;
1358
+ }
1359
+ .content-wrap {
1360
+ text-align: right;
1361
+ .robot-item {
1362
+ text-align: right;
1363
+ }
1364
+ .voice-bg {
1365
+ flex-direction: row-reverse;
1366
+ align-self: flex-end;
1367
+ img {
1368
+ transform: rotateY(180deg);
1369
+ }
1370
+ span {
1371
+ margin-left: 0;
1372
+ margin-right: 8px;
1373
+ text-align: right;
1374
+ }
1375
+ }
1376
+ }
1377
+ .content-wrap .content {
1378
+ background-color: #e7f0ff;
1379
+ border-radius: 8px 0px 8px 8px;
1380
+ border: 1px solid #e7f0ff;
1381
+ box-sizing: border-box;
1382
+ }
1383
+ .upload-image {
1384
+ margin-right: 8px;
1385
+ }
1386
+ }
1387
+ }
1388
+ .bot-message {
1389
+ display: block;
1390
+ text-decoration: underline;
1391
+ color: #5585f5;
1392
+ cursor: pointer;
1393
+ }
1394
+ .session-end {
1395
+ display: flex;
1396
+ justify-content: center;
1397
+ margin-top: 15px;
1398
+ margin-bottom: 25px;
1399
+ .end {
1400
+ display: flex;
1401
+ justify-content: center;
1402
+ width: 30%;
1403
+ height: 36px;
1404
+ line-height: 36px;
1405
+ background-color: #e8e8e8;
1406
+ color: #939393;
1407
+ font-size: 14px;
1408
+ border-radius: 18px;
1409
+ }
1410
+ }
1411
+ .menu-tip {
1412
+ font-size: 20px;
1413
+ align-self: flex-end;
1414
+ margin-bottom: 8px;
1415
+ background: #f6f6f6;
1416
+ height: 25px;
1417
+ line-height: 25px;
1418
+ width: 30px;
1419
+ border-radius: 4px;
1420
+ }
1421
+ .menu-item {
1422
+ :hover {
1423
+ background: red;
1424
+ }
1425
+ }
1426
+ .menu-main {
1427
+ width: 100px;
1428
+ background-color: #101627;
1429
+ color: #fff;
1430
+ text-align: left;
1431
+ border-radius: 6px;
1432
+ padding: 10px 0;
1433
+ z-index: 3000;
1434
+ cursor: pointer;
1435
+ .menu-item {
1436
+ padding: 0 10px;
1437
+ }
1438
+ .menu-item:hover {
1439
+ background-color: #efefef;
1440
+ color: #000;
1441
+ }
1442
+ }
1443
+ .first-name {
1444
+ width: 40px;
1445
+ height: 40px;
1446
+ line-height: 40px;
1447
+ text-align: center;
1448
+ background: #5585f5;
1449
+ color: #fff;
1450
+ border-radius: 100%;
1451
+ font-size: 15px;
1452
+ }
1453
+ .msg-avatar {
1454
+ width: 40px;
1455
+ height: 40px;
1456
+ border-radius: 50%;
1457
+
1458
+ object-fit: cover;
1459
+ background: #fff;
1460
+ }
1461
+ /deep/ .ant-spin-spinning {
1462
+ display: flex;
1463
+ align-items: center;
1464
+ }
1465
+ .system-msg {
1466
+ margin: 10px 0;
1467
+ text-align: left;
1468
+ line-height: 20px;
1469
+ word-wrap: break-word;
1470
+ display: flex;
1471
+ justify-content: center;
1472
+ .system-icon {
1473
+ font-size: 18px;
1474
+ color: #ffc200;
1475
+ vertical-align: middle;
1476
+ margin-right: 10px;
1477
+ margin-top: 1px;
1478
+ }
1479
+ .system-btn {
1480
+ color: @primary-color;
1481
+ margin-left: 8px;
1482
+ cursor: pointer;
1483
+ text-decoration: underline;
1484
+ }
1485
+ }
1486
+ }
1487
+ /deep/ p {
1488
+ margin-bottom: 0;
1489
+ }
1490
+ </style>