cnhis-design-vue 2.1.56 → 2.1.57

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 (574) hide show
  1. package/.prettierrc.js +27 -27
  2. package/CHANGELOG.md +2646 -2551
  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 +308 -308
  15. package/es/breadcrumb/index.js +8 -8
  16. package/es/button/index.js +31 -31
  17. package/es/calendar/index.js +8 -8
  18. package/es/captcha/index.js +7 -7
  19. package/es/card/index.js +8 -8
  20. package/es/carousel/index.js +8 -8
  21. package/es/cascader/index.js +8 -8
  22. package/es/checkbox/index.js +17 -17
  23. package/es/col/index.js +8 -8
  24. package/es/collapse/index.js +8 -8
  25. package/es/color-picker/index.js +5 -5
  26. package/es/comment/index.js +8 -8
  27. package/es/config-provider/index.js +8 -8
  28. package/es/date-picker/index.js +30 -30
  29. package/es/descriptions/index.js +8 -8
  30. package/es/divider/index.js +8 -8
  31. package/es/drag-layout/index.js +5 -5
  32. package/es/drawer/index.js +8 -8
  33. package/es/dropdown/index.js +8 -8
  34. package/es/editor/index.js +1 -1
  35. package/es/ellipsis/index.js +3211 -0
  36. package/es/ellipsis/style.css +1 -0
  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 +1600 -431
  42. package/es/form-table/style.css +1 -1
  43. package/es/grid/index.js +4 -4
  44. package/es/index/index.js +2567 -1411
  45. package/es/index/style.css +1 -1
  46. package/es/input/index.js +11 -11
  47. package/es/input-number/index.js +8 -8
  48. package/es/keep-cache/index.js +9 -9
  49. package/es/layout/index.js +8 -8
  50. package/es/list/index.js +8 -8
  51. package/es/locale-provider/index.js +8 -8
  52. package/es/map/index.js +13 -13
  53. package/es/mentions/index.js +8 -8
  54. package/es/menu/index.js +8 -8
  55. package/es/message/index.js +8 -8
  56. package/es/multi-chat/index.js +116 -116
  57. package/es/multi-chat-client/index.js +110 -110
  58. package/es/multi-chat-history/index.js +6 -6
  59. package/es/multi-chat-record/index.js +27 -27
  60. package/es/multi-chat-setting/index.js +43 -43
  61. package/es/multi-chat-sip/index.js +1 -1
  62. package/es/notification/index.js +8 -8
  63. package/es/page-header/index.js +8 -8
  64. package/es/pagination/index.js +8 -8
  65. package/es/popconfirm/index.js +8 -8
  66. package/es/popover/index.js +8 -8
  67. package/es/progress/index.js +8 -8
  68. package/es/radio/index.js +17 -17
  69. package/es/rate/index.js +8 -8
  70. package/es/result/index.js +8 -8
  71. package/es/row/index.js +8 -8
  72. package/es/scale-container/index.js +3356 -0
  73. package/es/scale-container/style.css +1 -0
  74. package/es/scale-view/index.js +158 -158
  75. package/es/select/index.js +83 -70
  76. package/es/select/style.css +1 -1
  77. package/es/select-label/index.js +50 -50
  78. package/es/select-person/index.js +20 -20
  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 +251 -238
  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 +4979 -1618
  105. package/lib/cui.umd.js +4979 -1618
  106. package/lib/cui.umd.min.js +67 -61
  107. package/package.json +110 -108
  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 +3144 -3144
  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 +385 -385
  132. package/packages/big-table/src/assets/style/table-global.less +175 -175
  133. package/packages/big-table/src/components/AutoLayoutButton.vue +279 -279
  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 +189 -189
  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 +103 -103
  171. package/packages/big-table/src/utils/format.js +557 -557
  172. package/packages/big-table/src/utils/nestTable.js +109 -109
  173. package/packages/big-table/src/utils/tableParse.js +234 -234
  174. package/packages/breadcrumb/index.js +21 -21
  175. package/packages/breadcrumb/src/Breadcrumb.jsx +11 -11
  176. package/packages/breadcrumb/src/BreadcrumbItem.jsx +11 -11
  177. package/packages/breadcrumb/src/BreadcrumbSeparator.jsx +11 -11
  178. package/packages/button/index.js +21 -21
  179. package/packages/button/src/Button.jsx +11 -11
  180. package/packages/button/src/ButtonGroup.jsx +11 -11
  181. package/packages/button/src/ButtonPrint/components/IdentityVerification.vue +181 -181
  182. package/packages/button/src/ButtonPrint/index.vue +766 -766
  183. package/packages/button/src/ButtonPrint/mixin/his-print.js +95 -95
  184. package/packages/calendar/index.js +12 -12
  185. package/packages/calendar/src/Calendar.jsx +11 -11
  186. package/packages/captcha/index.js +9 -9
  187. package/packages/captcha/src/Captcha.vue +164 -164
  188. package/packages/captcha/src/Index.vue +47 -47
  189. package/packages/captcha/src/SlideVerify.vue +285 -285
  190. package/packages/card/index.js +21 -21
  191. package/packages/card/src/Card.jsx +12 -12
  192. package/packages/card/src/CardGrid.js +7 -7
  193. package/packages/card/src/CardMeta.js +7 -7
  194. package/packages/carousel/index.js +12 -12
  195. package/packages/carousel/src/Carousel.jsx +12 -12
  196. package/packages/cascader/index.js +12 -12
  197. package/packages/cascader/src/Cascader.jsx +11 -11
  198. package/packages/checkbox/index.js +30 -30
  199. package/packages/checkbox/src/Checkbox.jsx +11 -11
  200. package/packages/checkbox/src/CheckboxImg/index.vue +141 -141
  201. package/packages/checkbox/src/Group.jsx +11 -11
  202. package/packages/col/index.js +13 -13
  203. package/packages/col/src/Col.jsx +11 -11
  204. package/packages/collapse/index.js +18 -18
  205. package/packages/collapse/src/Collapse.jsx +11 -11
  206. package/packages/collapse/src/Panel.jsx +11 -11
  207. package/packages/color-picker/index.js +10 -10
  208. package/packages/color-picker/src/color-picker.vue +191 -191
  209. package/packages/color-picker/src/style.less +109 -109
  210. package/packages/comment/index.js +12 -12
  211. package/packages/comment/src/Comment.jsx +11 -11
  212. package/packages/config-provider/index.js +12 -12
  213. package/packages/config-provider/src/ConfigProvider.jsx +11 -11
  214. package/packages/date-picker/index.js +26 -26
  215. package/packages/date-picker/src/DatePicker.jsx +12 -12
  216. package/packages/date-picker/src/MonthPicker.jsx +11 -11
  217. package/packages/date-picker/src/RangePicker.jsx +11 -11
  218. package/packages/date-picker/src/WeekPicker.jsx +11 -11
  219. package/packages/date-picker/src/utils/index.js +374 -374
  220. package/packages/descriptions/index.js +18 -18
  221. package/packages/descriptions/src/Descriptions.jsx +11 -11
  222. package/packages/descriptions/src/Item.jsx +11 -11
  223. package/packages/divider/index.js +12 -12
  224. package/packages/divider/src/Divider.jsx +11 -11
  225. package/packages/drag-layout/DragFormLeftItem.vue +173 -173
  226. package/packages/drag-layout/DragFormRightItem.vue +284 -284
  227. package/packages/drag-layout/I18n-mixins.js +10 -10
  228. package/packages/drag-layout/drag-layout.vue +778 -778
  229. package/packages/drag-layout/index.js +12 -12
  230. package/packages/drawer/index.js +12 -12
  231. package/packages/drawer/src/Drawer.jsx +11 -11
  232. package/packages/dropdown/index.js +12 -12
  233. package/packages/dropdown/src/Dropdown.jsx +11 -11
  234. package/packages/editor/index.js +9 -9
  235. package/packages/ellipsis/index.js +9 -0
  236. package/packages/ellipsis/src/Ellipsis.vue +65 -0
  237. package/packages/empty/index.js +12 -12
  238. package/packages/empty/src/Empty.jsx +11 -11
  239. package/packages/fabric-chart/index.js +9 -9
  240. package/packages/fabric-chart/src/FabricGrid.vue +66 -66
  241. package/packages/fabric-chart/src/components/DropPopup.vue +90 -90
  242. package/packages/fabric-chart/src/components/MouseRightClick.vue +168 -168
  243. package/packages/fabric-chart/src/components/TimeScaleValue.vue +115 -115
  244. package/packages/fabric-chart/src/const/defaultVaule.js +59 -59
  245. package/packages/fabric-chart/src/fabric-chart/FabricLines.vue +552 -552
  246. package/packages/fabric-chart/src/fabric-chart/FabricPolylines.vue +1126 -1126
  247. package/packages/fabric-chart/src/fabric-chart/FabricScaleValue.vue +134 -134
  248. package/packages/fabric-chart/src/fabric-chart/FabricTextGroup.vue +566 -566
  249. package/packages/fabric-chart/src/mixins/fabricCommon.js +95 -95
  250. package/packages/fabric-chart/src/mixins/fabricObject.js +193 -193
  251. package/packages/fabric-chart/src/mixins/type.js +5 -5
  252. package/packages/form/index.js +16 -16
  253. package/packages/form/src/Form.jsx +11 -11
  254. package/packages/form/src/Item.jsx +11 -11
  255. package/packages/form-model/index.js +14 -14
  256. package/packages/form-model/src/FormModel.jsx +11 -11
  257. package/packages/form-model/src/Item.jsx +11 -11
  258. package/packages/form-table/index.js +16 -10
  259. package/packages/form-table/src/FormTable.vue +1110 -1110
  260. package/packages/form-table/src/components/table-component/config-data/index.js +80 -80
  261. package/packages/form-table/src/components/table-component/global-props.js +22 -22
  262. package/packages/form-table/src/components/table-component/index.js +8 -8
  263. package/packages/form-table/src/components/table-component/mixins/bound-date.js +455 -455
  264. package/packages/form-table/src/components/table-component/mixins/dateType.js +217 -217
  265. package/packages/form-table/src/components/table-component/mixins/item-default.js +257 -257
  266. package/packages/form-table/src/components/table-component/mixins/search.js +1238 -1228
  267. package/packages/form-table/src/components/table-component/register-com.js +30 -30
  268. package/packages/form-table/src/components/table-component/table-age/table-age.vue +175 -175
  269. package/packages/form-table/src/components/table-component/table-date-picker/table-date-picker.vue +87 -87
  270. package/packages/form-table/src/components/table-component/table-digital/table-digital.vue +93 -93
  271. package/packages/form-table/src/components/table-component/table-input/table-input.vue +81 -81
  272. package/packages/form-table/src/components/table-component/table-input-password/table-input-password.vue +126 -126
  273. package/packages/form-table/src/components/table-component/table-month-picker/table-month-picker.vue +55 -55
  274. package/packages/form-table/src/components/table-component/table-search/table-search.vue +174 -128
  275. package/packages/form-table/src/components/table-component/table-search-more/table-search-more.vue +191 -144
  276. package/packages/form-table/src/components/table-component/table-select/table-select.vue +64 -64
  277. package/packages/form-table/src/components/table-component/table-select-multiple/table-select-multiple.vue +81 -81
  278. package/packages/form-table/src/components/table-component/table-textarea/table-textarea.vue +76 -76
  279. package/packages/form-table/src/components/table-component/table-time-picker/table-time-picker.vue +55 -55
  280. package/packages/form-table/src/components/table-component/table-tree-select/table-tree-select.vue +135 -135
  281. package/packages/form-table/src/components/table-component/text-over-tooltip/TextOverTooltip.vue +97 -97
  282. package/packages/form-table/src/components/table-item/form-event.js +81 -81
  283. package/packages/form-table/src/components/table-item/global-props.js +27 -27
  284. package/packages/form-table/src/components/table-item/index.js +4 -4
  285. package/packages/form-table/src/components/table-item/render-methods.js +28 -28
  286. package/packages/form-table/src/components/table-item/validate-rules.js +520 -520
  287. package/packages/form-table/src/components/table-mixins/form-commom.js +98 -98
  288. package/packages/form-table/src/components/table-mixins/index.js +3 -3
  289. package/packages/form-table/src/components/table-utils/index.js +112 -112
  290. package/packages/form-table/src/disabledDetail.less +46 -46
  291. package/packages/grid/index.js +10 -10
  292. package/packages/grid/src/grid.js +28 -28
  293. package/packages/icon/index.js +10 -10
  294. package/packages/icon/src/icon.js +13 -13
  295. package/packages/index.js +313 -307
  296. package/packages/input/index.js +30 -30
  297. package/packages/input/src/Group.jsx +11 -11
  298. package/packages/input/src/Input.jsx +11 -11
  299. package/packages/input/src/Password.jsx +11 -11
  300. package/packages/input/src/Search.jsx +11 -11
  301. package/packages/input/src/TextArea.jsx +11 -11
  302. package/packages/input/src/input-quick/components/quick-item.vue +284 -284
  303. package/packages/input/src/input-quick/components/quick-popover.vue +596 -596
  304. package/packages/input/src/input-quick/index.vue +137 -137
  305. package/packages/input-number/index.js +12 -12
  306. package/packages/input-number/src/InputNumber.jsx +11 -11
  307. package/packages/keep-cache/KeepCache.js +236 -236
  308. package/packages/keep-cache/index.css +2 -2
  309. package/packages/keep-cache/index.js +8 -8
  310. package/packages/layout/index.js +27 -27
  311. package/packages/layout/src/Content.jsx +11 -11
  312. package/packages/layout/src/Footer.jsx +11 -11
  313. package/packages/layout/src/Header.jsx +11 -11
  314. package/packages/layout/src/Layout.jsx +11 -11
  315. package/packages/layout/src/Sider.jsx +11 -11
  316. package/packages/list/index.js +21 -21
  317. package/packages/list/src/Item.jsx +11 -11
  318. package/packages/list/src/ItemMeta.jsx +11 -11
  319. package/packages/list/src/List.jsx +11 -11
  320. package/packages/locale-provider/index.js +12 -12
  321. package/packages/locale-provider/src/LocaleProvider.jsx +11 -11
  322. package/packages/map/index.js +9 -9
  323. package/packages/map/src/Map.vue +484 -484
  324. package/packages/map/src/popup-map.vue +53 -53
  325. package/packages/mentions/index.js +18 -18
  326. package/packages/mentions/src/Mentions.jsx +11 -11
  327. package/packages/mentions/src/Option.jsx +11 -11
  328. package/packages/menu/index.js +27 -27
  329. package/packages/menu/src/Divider.jsx +11 -11
  330. package/packages/menu/src/Item.jsx +11 -11
  331. package/packages/menu/src/ItemGroup.jsx +11 -11
  332. package/packages/menu/src/Menu.jsx +11 -11
  333. package/packages/menu/src/SubMenu.jsx +11 -11
  334. package/packages/message/index.js +8 -8
  335. package/packages/modal/index.js +10 -10
  336. package/packages/modal/src/Modal.js +7 -7
  337. package/packages/multi-chat/chat/addConference.vue +200 -200
  338. package/packages/multi-chat/chat/addMembers.vue +411 -411
  339. package/packages/multi-chat/chat/advancedFilter.vue +372 -372
  340. package/packages/multi-chat/chat/calling.vue +246 -246
  341. package/packages/multi-chat/chat/chatFooter.vue +1596 -1596
  342. package/packages/multi-chat/chat/chatHistory.vue +605 -605
  343. package/packages/multi-chat/chat/chatMain.vue +1486 -1486
  344. package/packages/multi-chat/chat/client/index.vue +149 -149
  345. package/packages/multi-chat/chat/delay.vue +177 -177
  346. package/packages/multi-chat/chat/evaluate.vue +343 -343
  347. package/packages/multi-chat/chat/messageRecord.vue +324 -324
  348. package/packages/multi-chat/chat/mixins/NoData.js +20 -20
  349. package/packages/multi-chat/chat/mixins/base.js +97 -97
  350. package/packages/multi-chat/chat/mixins/uniRTCAPI.js +80 -80
  351. package/packages/multi-chat/chat/mixins/viewerOptions.js +67 -67
  352. package/packages/multi-chat/chat/quickReply.vue +439 -439
  353. package/packages/multi-chat/chat/robot/index.vue +312 -312
  354. package/packages/multi-chat/chat/scrollList.vue +1238 -1238
  355. package/packages/multi-chat/chat/videoVoiceList.vue +348 -348
  356. package/packages/multi-chat/chat/voice.vue +431 -431
  357. package/packages/multi-chat/components/avatar.vue +113 -113
  358. package/packages/multi-chat/components/chat-tabs-header.vue +251 -251
  359. package/packages/multi-chat/components/classify-tabs.vue +185 -185
  360. package/packages/multi-chat/components/empty.vue +24 -24
  361. package/packages/multi-chat/components/modal-refuse-reason.vue +112 -112
  362. package/packages/multi-chat/components/modal-sip.vue +160 -160
  363. package/packages/multi-chat/components/modal-user-transfer.vue +98 -98
  364. package/packages/multi-chat/components/msg-describe.vue +138 -138
  365. package/packages/multi-chat/components/msg-picture.vue +68 -68
  366. package/packages/multi-chat/components/msg-prescription.vue +205 -205
  367. package/packages/multi-chat/components/read-record.vue +133 -133
  368. package/packages/multi-chat/components/read-status.vue +34 -34
  369. package/packages/multi-chat/components/user-status.vue +198 -198
  370. package/packages/multi-chat/index.js +7 -7
  371. package/packages/multi-chat/setting/authority/index.vue +156 -156
  372. package/packages/multi-chat/setting/authority/roleSetting.vue +204 -204
  373. package/packages/multi-chat/setting/baseInfo/index.vue +1316 -1316
  374. package/packages/multi-chat/setting/customerService/batchSelect.vue +403 -403
  375. package/packages/multi-chat/setting/customerService/index.vue +273 -273
  376. package/packages/multi-chat/setting/event/edit/condition.vue +128 -128
  377. package/packages/multi-chat/setting/event/edit/index.vue +437 -437
  378. package/packages/multi-chat/setting/event/edit/notice.vue +129 -129
  379. package/packages/multi-chat/setting/event/edit/strategy.vue +98 -98
  380. package/packages/multi-chat/setting/event/index.vue +249 -249
  381. package/packages/multi-chat/setting/index.vue +269 -269
  382. package/packages/multi-chat/setting/page.vue +14 -14
  383. package/packages/multi-chat/setting/sessionList/index.vue +412 -412
  384. package/packages/multi-chat/setting/sessionList/messageRecord.vue +372 -372
  385. package/packages/multi-chat/setting/userConfig/index.vue +124 -124
  386. package/packages/multi-chat/setting/worktime/index.vue +274 -274
  387. package/packages/multi-chat/store/actions.js +452 -452
  388. package/packages/multi-chat/store/getters.js +371 -371
  389. package/packages/multi-chat/store/helper.js +66 -66
  390. package/packages/multi-chat/store/index.js +50 -50
  391. package/packages/multi-chat/store/mutation.js +296 -296
  392. package/packages/multi-chat/store/state.js +117 -117
  393. package/packages/multi-chat/style/emoji.css +315 -315
  394. package/packages/multi-chat/style/message.mixin.less +38 -38
  395. package/packages/multi-chat/utils/chatSock.js +93 -93
  396. package/packages/multi-chat/utils/compressImage.js +115 -115
  397. package/packages/multi-chat/utils/emoji.json +68 -68
  398. package/packages/multi-chat/utils/index.js +259 -259
  399. package/packages/multi-chat/utils/observer-scroll.js +49 -49
  400. package/packages/multi-chat/utils/panelsetting.js +48 -48
  401. package/packages/multi-chat-client/index.js +7 -7
  402. package/packages/multi-chat-history/index.js +7 -7
  403. package/packages/multi-chat-record/index.js +7 -7
  404. package/packages/multi-chat-setting/index.js +7 -7
  405. package/packages/multi-chat-sip/index.js +6 -6
  406. package/packages/notification/index.js +8 -8
  407. package/packages/page-header/index.js +12 -12
  408. package/packages/page-header/src/PageHeader.jsx +11 -11
  409. package/packages/pagination/index.js +12 -12
  410. package/packages/pagination/src/Pagination.jsx +11 -11
  411. package/packages/popconfirm/index.js +12 -12
  412. package/packages/popconfirm/src/Popconfirm.jsx +11 -11
  413. package/packages/popover/index.js +12 -12
  414. package/packages/popover/src/Popover.jsx +11 -11
  415. package/packages/progress/index.js +12 -12
  416. package/packages/progress/src/Progress.jsx +11 -11
  417. package/packages/radio/index.js +33 -33
  418. package/packages/radio/src/Group.jsx +11 -11
  419. package/packages/radio/src/Radio.jsx +11 -11
  420. package/packages/radio/src/RadioButton.jsx +11 -11
  421. package/packages/radio/src/RadioImg/index.vue +124 -124
  422. package/packages/rate/index.js +12 -12
  423. package/packages/rate/src/Rate.jsx +11 -11
  424. package/packages/result/index.js +12 -12
  425. package/packages/result/src/Result.jsx +11 -11
  426. package/packages/row/index.js +12 -12
  427. package/packages/row/src/Row.jsx +11 -11
  428. package/packages/scale-container/index.js +9 -0
  429. package/packages/scale-container/src/ScaleContainer.vue +194 -0
  430. package/packages/scale-container/src/scale.png +0 -0
  431. package/packages/scale-view/NoData.vue +81 -81
  432. package/packages/scale-view/answerParse.vue +133 -133
  433. package/packages/scale-view/customList.vue +801 -801
  434. package/packages/scale-view/data.js +80 -80
  435. package/packages/scale-view/evaluateCountdown.vue +155 -155
  436. package/packages/scale-view/evaluatePage.vue +202 -202
  437. package/packages/scale-view/formitem/data.js +3991 -3991
  438. package/packages/scale-view/formitem/index.js +6 -6
  439. package/packages/scale-view/formitem/r-address.vue +238 -238
  440. package/packages/scale-view/formitem/r-choice.vue +726 -726
  441. package/packages/scale-view/formitem/r-input.vue +92 -92
  442. package/packages/scale-view/formitem/r-prompt.vue +52 -52
  443. package/packages/scale-view/formitem/r-time.vue +285 -285
  444. package/packages/scale-view/formitem/r-upload-custom-list.vue +242 -242
  445. package/packages/scale-view/formitem/r-upload.vue +287 -287
  446. package/packages/scale-view/formitem/text-over-tooltip/TextOverTooltip.vue +98 -98
  447. package/packages/scale-view/index.js +17 -17
  448. package/packages/scale-view/mixin/NoData.js +38 -38
  449. package/packages/scale-view/mixin/evaluate.js +146 -146
  450. package/packages/scale-view/mixin/index.js +337 -337
  451. package/packages/scale-view/mixin/judgeTypes.js +267 -267
  452. package/packages/scale-view/scaleView.vue +2010 -2010
  453. package/packages/select/index.js +27 -27
  454. package/packages/select/src/CustomSelect/index.vue +130 -130
  455. package/packages/select/src/OptGroup.jsx +11 -11
  456. package/packages/select/src/Option.jsx +11 -11
  457. package/packages/select/src/Select/Select.vue +231 -231
  458. package/packages/select/src/Select/index.js +12 -12
  459. package/packages/select/src/TableSelect/index.vue +514 -514
  460. package/packages/select-label/index.js +14 -14
  461. package/packages/select-label/label-classify.vue +129 -129
  462. package/packages/select-label/labelFormContent.vue +787 -787
  463. package/packages/select-label/select-label.vue +581 -581
  464. package/packages/select-person/index.js +10 -10
  465. package/packages/select-person/search-tree.vue +373 -373
  466. package/packages/select-person/select-person.vue +1696 -1696
  467. package/packages/shortcut-setter/index.js +12 -12
  468. package/packages/shortcut-setter/src/ShortcutSetter.vue +55 -55
  469. package/packages/shortcut-setter/src/ShortcutSetterItem.vue +84 -84
  470. package/packages/shortcut-setter/src/utils/index.js +63 -63
  471. package/packages/skeleton/index.js +12 -12
  472. package/packages/skeleton/src/Skeleton.jsx +11 -11
  473. package/packages/slider/index.js +12 -12
  474. package/packages/slider/src/Slider.jsx +11 -11
  475. package/packages/space/index.js +12 -12
  476. package/packages/space/src/Space.jsx +11 -11
  477. package/packages/spin/index.js +12 -12
  478. package/packages/spin/src/Spin.jsx +11 -11
  479. package/packages/statistic/index.js +18 -18
  480. package/packages/statistic/src/Countdown.jsx +11 -11
  481. package/packages/statistic/src/Statistic.jsx +11 -11
  482. package/packages/steps/index.js +18 -18
  483. package/packages/steps/src/Step.jsx +11 -11
  484. package/packages/steps/src/Steps.jsx +11 -11
  485. package/packages/switch/index.js +12 -12
  486. package/packages/switch/src/Switch.jsx +11 -11
  487. package/packages/table-filter/index.js +27 -27
  488. package/packages/table-filter/src/base-search-com/BaseSearch.vue +2572 -2572
  489. package/packages/table-filter/src/classification/Classification-com.vue +1756 -1756
  490. package/packages/table-filter/src/classification/search-class-name.vue +266 -266
  491. package/packages/table-filter/src/classification/search-professional-model.vue +680 -680
  492. package/packages/table-filter/src/components/TextOverTooltip.vue +107 -107
  493. package/packages/table-filter/src/components/age-com/index.vue +205 -205
  494. package/packages/table-filter/src/components/button-group/ButtonGroup.vue +162 -162
  495. package/packages/table-filter/src/components/button-icon/button-icon.js +33 -33
  496. package/packages/table-filter/src/components/button-icon/getBtnIcon.js +34 -34
  497. package/packages/table-filter/src/components/c-tree-select/tree-select.vue +336 -336
  498. package/packages/table-filter/src/components/drop-button/drop-button.vue +224 -224
  499. package/packages/table-filter/src/components/drop-button/head-btn-icon.js +33 -33
  500. package/packages/table-filter/src/components/drop-view/drop-view.vue +89 -89
  501. package/packages/table-filter/src/components/multi-select/multi-select.vue +227 -227
  502. package/packages/table-filter/src/components/out-quick-search/out-quick-search.vue +340 -340
  503. package/packages/table-filter/src/components/range-age/index.vue +172 -172
  504. package/packages/table-filter/src/components/search-condition/SearchCondition.vue +1897 -1897
  505. package/packages/table-filter/src/components/search-condition/fieldTypeList.js +169 -169
  506. package/packages/table-filter/src/components/search-filter/SearchFilter.vue +278 -278
  507. package/packages/table-filter/src/components/search-modal/set-classification.vue +310 -310
  508. package/packages/table-filter/src/components/table-modal/TableModal.vue +463 -463
  509. package/packages/table-filter/src/const/dataOptions.js +43 -43
  510. package/packages/table-filter/src/const/index.js +1 -1
  511. package/packages/table-filter/src/index.vue +584 -584
  512. package/packages/table-filter/src/mixins/mixins.js +694 -694
  513. package/packages/table-filter/src/mixins/tableSearchCon.js +128 -128
  514. package/packages/table-filter/src/mixins/wordBookutils.js +102 -102
  515. package/packages/table-filter/src/quick-search/QuickSearch.vue +2125 -2125
  516. package/packages/tabs/index.js +18 -18
  517. package/packages/tabs/src/TabPane.jsx +11 -11
  518. package/packages/tabs/src/Tabs.jsx +11 -11
  519. package/packages/tag/index.js +21 -21
  520. package/packages/tag/src/CheckableTag.jsx +11 -11
  521. package/packages/tag/src/Tag.jsx +11 -11
  522. package/packages/tag/src/TagGroup.vue +621 -621
  523. package/packages/time-picker/index.js +12 -12
  524. package/packages/time-picker/src/TimePicker.jsx +11 -11
  525. package/packages/timeline/index.js +14 -14
  526. package/packages/timeline/src/Item.jsx +11 -11
  527. package/packages/timeline/src/Timeline.jsx +11 -11
  528. package/packages/tooltip/index.js +12 -12
  529. package/packages/tooltip/src/Tooltip.jsx +11 -11
  530. package/packages/transfer/index.js +12 -12
  531. package/packages/transfer/src/Transfer.jsx +11 -11
  532. package/packages/tree/index.js +18 -18
  533. package/packages/tree/src/Tree.jsx +11 -11
  534. package/packages/tree/src/TreeNode.jsx +11 -11
  535. package/packages/tree-select/index.js +18 -18
  536. package/packages/tree-select/src/TreeNode.jsx +11 -11
  537. package/packages/tree-select/src/TreeSelect.jsx +11 -11
  538. package/packages/upload/chunk-upload/chunk-upload-new.vue +1001 -1001
  539. package/packages/upload/chunk-upload/vod-chunk-upload.vue +749 -749
  540. package/packages/upload/chunk-upload/vod-upload-modal.vue +100 -100
  541. package/packages/upload/index.js +12 -12
  542. package/packages/upload/src/Upload.jsx +11 -11
  543. package/packages/verification-code/SlideVerify.vue +306 -306
  544. package/packages/verification-code/index.js +17 -17
  545. package/packages/verification-code/verification-code.vue +147 -147
  546. package/src/assets/img/nodata.png +0 -0
  547. package/src/component/player-vod/index.vue +57 -57
  548. package/src/component/player-vod/player.vue +188 -188
  549. package/src/component/player-vod/video-list.vue +262 -262
  550. package/src/component/player-vod/video-modal.vue +128 -128
  551. package/src/component/select-options/index.vue +413 -0
  552. package/src/component/select-pages/index.vue +95 -74
  553. package/src/component/svg/index.vue +59 -59
  554. package/src/core/create.js +6 -6
  555. package/src/core/event.js +23 -23
  556. package/src/core/table-methods.js +444 -444
  557. package/src/directive/flexibleResize.js +151 -151
  558. package/src/directive/preventReClick.js +12 -12
  559. package/src/directive/scroll.js +230 -230
  560. package/src/global/variable.js +2 -2
  561. package/src/style/normalize.css +424 -424
  562. package/src/style/style.less +49 -49
  563. package/src/utils/UniRTCv2.js +626 -626
  564. package/src/utils/chatFetch.js +61 -61
  565. package/src/utils/clickoutside.js +75 -75
  566. package/src/utils/crypto.js +25 -25
  567. package/src/utils/index.js +81 -81
  568. package/src/utils/kty-sdk.js +582 -582
  569. package/src/utils/kty.min-1.0.0.js +14378 -14378
  570. package/src/utils/sip-device.js +79 -79
  571. package/src/utils/time-domain.js +193 -193
  572. package/src/utils/trtc.js +1 -1
  573. package/src/utils/utils-map.js +484 -484
  574. package/src/utils/vexutils.js +836 -836
@@ -1,1596 +1,1596 @@
1
- <template>
2
- <a-layout-footer style="position: relative;cursor:pointer" v-show="!queueItem || !hideQueueBtns">
3
- <div class="message-operate">
4
- <a-upload
5
- v-if="isShow('PICTURE')"
6
- class="operate-icon"
7
- v-decorator="[
8
- 'upload',
9
- {
10
- valuePropName: 'fileList'
11
- }
12
- ]"
13
- :beforeUpload="beforeUpload"
14
- >
15
- <a-tooltip placement="top">
16
- <template slot="title">{{ i18nText('1.9.181') }}</template>
17
- <svg-icon icon-class="liaotiantupian1" class="toolbar"></svg-icon>
18
- </a-tooltip>
19
- </a-upload>
20
- <a-popover :title="i18nText('1.9.354')" trigger="click" v-model="visible">
21
- <a slot="content">
22
- <div class="emoji-wrapper">
23
- <p v-for="(item, index) in emojiList" :key="index" @click="() => handleEmojiClick(item)" :class="['twa', 'emoji', item]"></p>
24
- </div>
25
- </a>
26
-
27
- <a-tooltip placement="top">
28
- <template slot="title">{{ i18nText('1.9.354') }}</template>
29
- <svg-icon icon-class="liaotianbiaoqing1" @click="showEmoji" class="toolbar" v-if="isShow('SMILINEFACE')"></svg-icon>
30
- </a-tooltip>
31
- </a-popover>
32
- <template v-for="(item, index) in toolbar">
33
- <a-tooltip placement="top" :key="index">
34
- <template slot="title">{{ item.name }}</template>
35
- <svg-icon :icon-class="item.icon" class="toolbar" :class="{ 'closed-toolbar': closedSession }" @click="handleToolBarClick(item)"></svg-icon>
36
- </a-tooltip>
37
- </template>
38
- <!-- 快捷回复 -->
39
- <a-tooltip placement="top" :title="'快捷回复'">
40
- <a-popover trigger="click" overlayClassName="quick-reply" placement="topLeft" v-model="quickShow">
41
- <template slot="content">
42
- <quick-reply :quickShow.sync="quickShow" :quickData="quickData"></quick-reply>
43
- </template>
44
- <svg-icon icon-class="liaotiankuaijiehuifu1" class="toolbar" @click="openQuickData" v-if="isShow('FASTREPLY') && isServer"></svg-icon>
45
- </a-popover>
46
- </a-tooltip>
47
- <a-tooltip :title="'语音通话'">
48
- <svg-icon icon-class="liaotianyuyindianhua" @click="handleVideoClick('voice')" class="toolbar" v-if="isShow('VOICE_CALL')"></svg-icon>
49
- <!-- <img
50
- width="20"
51
- src="../img/multi-voice.png"
52
- @click="handleVideoClick('voice')"
53
- v-if="isShow('VOICE_CALL') && isServer"
54
- /> -->
55
- </a-tooltip>
56
- <a-tooltip :title="'发送语音'">
57
- <svg-icon icon-class="liaotianyuyin1" @click="handleVoice(true)" class="toolbar" :class="{ 'click-voice': showVoice }" v-if="isShow('VOICE')"></svg-icon>
58
- <!-- <img
59
- width="20"
60
- src="../img/multi-video.png"
61
- @click="handleVideoClick('video')"
62
- v-if="isShow('VIDEO') && isServer"
63
- /> -->
64
- </a-tooltip>
65
- <a-tooltip :title="'视频通话'">
66
- <svg-icon icon-class="liaotianshipin1" @click="handleVideoClick('video')" class="toolbar" v-if="isShow('VIDEO')"></svg-icon>
67
- <!-- <img
68
- width="20"
69
- src="../img/multi-video.png"
70
- @click="handleVideoClick('video')"
71
- v-if="isShow('VIDEO') && isServer"
72
- /> -->
73
- </a-tooltip>
74
- </div>
75
- <div class="message" style="text-align: left">
76
- <div class="content-edit">
77
- <div
78
- ref="contentEdit"
79
- @click="saveRange"
80
- @keyup.exact="saveRange"
81
- @change="changeData"
82
- @input="handleInput"
83
- @compositionstart="() => (this.ping = true)"
84
- @compositionend="() => (this.ping = false)"
85
- @keydown="handleKeyDown"
86
- @paste="handlePaste"
87
- contenteditable="true"
88
- ></div>
89
- </div>
90
- <span class="prompt">Enter {{ i18nText('1.2.1.11.6') }}, Ctrl + Enter {{ i18nText('1.9.356') }}</span>
91
- <a-button :disabled="sendDisabled" class="message-send" @click="handleMsgSend">{{ i18nText('1.2.1.11.6') }}</a-button>
92
- </div>
93
- <div id="canvas-wrapper" style="display: none">
94
- <canvas id="canvas"></canvas>
95
- </div>
96
- <template v-if="showOverlay">
97
- <div class="overlay" v-if="(!onChating || sessionEnd) && !queueItem"></div>
98
- </template>
99
- <div class="queue-page" v-if="queueItem">
100
- <div v-if="accessDesc" class="queue-tips">{{ accessDesc }}</div>
101
- <div class="queue-btns">
102
- <div v-if="enableTransfer" class="toSession" @click="handleOpenTransfer">
103
- {{ i18nText('1.10.201') }}
104
- </div>
105
- <div class="toSession primary" @click="handleAccess">
106
- {{ accessText || i18nText('1.9.352') }}
107
- </div>
108
- <div v-if="enableRefuse" class="toSession danger" @click="openRefuseReason">
109
- {{ i18nText('1.9.353') }}
110
- </div>
111
- </div>
112
- </div>
113
- <!-- 操作栏弹窗 -->
114
- <a-modal
115
- :wrapClassName="isFullScreen ? 'chat-footer-modal-big' : 'chat-footer-modal-small'"
116
- :title="modalData.name"
117
- :visible="modalShow"
118
- :maskClosable="false"
119
- :mask="true"
120
- :width="isFullScreen ? '100%' : '80%'"
121
- destroyOnClose
122
- :okText="i18nText('1.2.1.11.6')"
123
- :footer="null"
124
- @cancel="handleClose"
125
- @ok="handleOk"
126
- >
127
- <iframe id="toolbarIframe" v-if="modalData.targetType === 'LINK_ADDRESS'" width="100%" height="100%" frameborder="0" :src="modalData.address" allow="camera;midi"></iframe>
128
- </a-modal>
129
- <!-- 接入拦截 -->
130
- <a-modal
131
- :title="accessInterceptTitle"
132
- :visible="accessIntercept"
133
- :maskClosable="false"
134
- :mask="true"
135
- :footer="null"
136
- destroyOnClose
137
- :width="600"
138
- :bodyStyle="{ height: '280px' }"
139
- @cancel="
140
- () => {
141
- this.accessIntercept = false;
142
- }
143
- "
144
- >
145
- <iframe width="100%" height="100%" frameborder="0" :src="accessInterceptUrl"></iframe>
146
- </a-modal>
147
- <!-- 语音消息 -->
148
- <voice v-if="showVoice" :showVoice="showVoice" @closeVoice="handleVoice"></voice>
149
- <ModalUserTransfer v-if="enableTransfer" :title="i18nText('1.10.201')" :visible.sync="transferVisible" :assemblyId="assemblyId" @ok="handleTransfer" />
150
- <ModalRefuseReason v-if="refuseVisible" :visible.sync="refuseVisible" :reasonList="refuseReasonList" @ok="handleRefuse" />
151
- </a-layout-footer>
152
- </template>
153
-
154
- <script>
155
- import { Layout, Tooltip, Upload, Modal, Popover, Button } from 'ant-design-vue';
156
- import SvgIcon from '@/component/svg/index.vue';
157
- import ImageCompress from '../utils/compressImage';
158
- import { parseParams } from '../utils/panelsetting';
159
-
160
- import { mapState, mapGetters, mapMutations, mapActions } from '../store/helper';
161
- import fetch, { qs } from '@/utils/chatFetch';
162
- import QuickReply from './quickReply';
163
- import voice from './voice';
164
- import ModalUserTransfer from '../components/modal-user-transfer';
165
- import ModalRefuseReason from '../components/modal-refuse-reason';
166
- import vexutils from '@/utils/vexutils';
167
- export default {
168
- inject: ['store', 'i18nText', 'dispatchEvent', 'registerEvent', 'unregisterEvent'],
169
- components: {
170
- AModal: Modal,
171
- [Layout.Footer.name]: Layout.Footer,
172
- [Tooltip.name]: Tooltip,
173
- [Upload.name]: Upload,
174
- [Button.name]: Button,
175
- [Popover.name]: Popover,
176
- SvgIcon,
177
- ModalRefuseReason,
178
- ModalUserTransfer,
179
- QuickReply,
180
- voice
181
- },
182
- props: {
183
- // 隐藏排队状态底部信息
184
- hideQueueBtns: {
185
- type: Boolean,
186
- default: false
187
- },
188
- validTalkingEnv: {
189
- type: Function
190
- },
191
- curChatType: {
192
- type: String
193
- },
194
- clientFinish: {
195
- type: Boolean,
196
- default: true
197
- },
198
- addvisible: {
199
- type: Boolean,
200
- default: false
201
- },
202
- isShowVoice: {
203
- type: Boolean,
204
- default: false
205
- }
206
- },
207
- data() {
208
- return {
209
- // message: "",
210
- visible: false,
211
- modalShow: false,
212
- modalData: {},
213
- ping: false,
214
- templateId: null, // 卡片模板
215
- isFullScreen: false,
216
- emojiList: require('../utils/emoji.json'),
217
- quickData: [],
218
- quickShow: false,
219
- screenshot: false,
220
- screenshotFile: null,
221
- timer: null,
222
- accessIntercept: false,
223
- curToolbarItem: {},
224
- showVoice: false,
225
- transferVisible: false,
226
- refuseVisible: false,
227
-
228
- apiResult: {}, // 前置条件的返回值
229
- };
230
- },
231
- computed: {
232
- ...mapGetters([
233
- 'userInfo',
234
- 'assemblyId',
235
- 'sessionId',
236
- 'serviceId',
237
- 'onChating',
238
- 'roomId',
239
- 'isServer',
240
- 'assemblySetting',
241
- 'chatType',
242
- 'sessionEnd',
243
- 'closedSession',
244
- 'clientParams',
245
- 'footerMessage',
246
- 'queueItem',
247
- 'sessionType',
248
- 'groupMembers',
249
- 'curScrollItem'
250
- ]),
251
- ...mapState({
252
- classify: state => state.lastClassify || state.classify
253
- }),
254
- message() {
255
- return this.footerMessage;
256
- },
257
- sendDisabled() {
258
- return this.message.length === 0;
259
- },
260
- orgId() {
261
- return this.userInfo?.sysParams?.orgId;
262
- },
263
- userId() {
264
- return this.userInfo?.sysParams?.userId;
265
- },
266
- toolbar() {
267
- if (!this.isServer) return [];
268
- let toolbar = this.assemblySetting.toolbar || [];
269
- toolbar = toolbar.filter(item => {
270
- const { source, isChecked, isCloseChecked, isClosedAble } = item;
271
- if (source === 'PC' && (
272
- (!this.closedSession && isChecked === 'Y') ||
273
- (this.closedSession && isCloseChecked === 'Y' && isChecked === 'Y' && isClosedAble === 'Y')
274
- )
275
- ){
276
- const showCondition = item.showCondition || [];
277
- const i = showCondition.findIndex(({ p_name, p_value }) => {
278
- const params = this.clientParams;
279
- const value = params[p_name];
280
- if (p_value.startsWith('form.')) {
281
- return value !== p_name[p_value.slice(5)];
282
- }
283
- return value !== p_value;
284
- });
285
- return i < 0;
286
- }
287
- return false;
288
- });
289
- return toolbar;
290
- },
291
- isListClassify() {
292
- return this.assemblySetting.isListClassify === 'Y';
293
- },
294
- showOverlay() {
295
- return this.curChatType != 'robot' || this.clientFinish;
296
- },
297
- accessText() {
298
- let obj = this.assemblySetting.serverSetting.serverInfo.filter(item => item.key === 'ongoingButton')[0];
299
- return obj?.value;
300
- },
301
- accessDesc() {
302
- let obj = this.assemblySetting.serverSetting.serverInfo.filter(item => item.key === 'ongoingDescription')[0];
303
- return obj?.value;
304
- },
305
- accessInterceptObj() {
306
- let obj = this.assemblySetting.accessInterceptSetting?.filter(item => item.source === 'PC');
307
- if (obj && obj[0]) {
308
- return obj[0];
309
- }
310
- return undefined;
311
- },
312
- // 拦截url
313
- accessInterceptUrl() {
314
- const { accessInterceptObj: obj, apiResult } = this;
315
- if (obj) {
316
- return this.getLinkAddress(obj,{result:apiResult});
317
- }
318
- return '';
319
- },
320
- // 拦截title
321
- accessInterceptTitle() {
322
- const { accessInterceptObj: obj } = this;
323
- if (obj) {
324
- let res = obj.titleI18n ? this.i18nText(obj.titleI18n) : obj.title;
325
- return res || '接入拦截';
326
- }
327
- return '接入拦截';
328
- },
329
- // 是否开启 排队转接
330
- enableTransfer() {
331
- if (this.assemblySetting.type === 'single') {
332
- let setting = null;
333
- if (this.isListClassify) {
334
- const classify = this.classify;
335
- const { listClassify = [] } = this.assemblySetting;
336
- const target = listClassify.find(item => item.classify === classify);
337
- setting = target ? target.serverFunctionSetting : null;
338
- } else {
339
- setting = this.assemblySetting?.serverSetting?.functionSetting;
340
- }
341
- if (setting) {
342
- const v = setting.find(item => item.function === 'TRANSFER_BEFORE');
343
- return v ? v.isChecked == 'Y' : false;
344
- }
345
- }
346
- return false;
347
- },
348
- enableRefuse() {
349
- let setting = null;
350
- if (this.isListClassify) {
351
- const classify = this.classify;
352
- const { listClassify = [] } = this.assemblySetting;
353
- const target = listClassify.find(item => item.classify === classify);
354
- setting = target ? target.serverFunctionSetting : null;
355
- } else {
356
- setting = this.assemblySetting?.serverSetting?.functionSetting;
357
- }
358
- if (setting) {
359
- const v = setting.find(item => item.function === 'ACCESS_REFUSE');
360
- return v ? v.isChecked == 'Y' : false;
361
- }
362
- return false;
363
- },
364
- refuseReasonList() {
365
- const list = this.assemblySetting?.refuseReasonSetting?.list || [];
366
- return list.filter(v => !!v.text);
367
- }
368
- },
369
-
370
- mounted() {
371
- this.bindEvents();
372
- },
373
- methods: {
374
- ...mapMutations([
375
- 'setLastClassify',
376
- 'setSessionTimer',
377
- 'setScrollQueueId',
378
- 'setCurScrollItem',
379
- 'setIsAppendMsg',
380
- 'setMessage',
381
- 'setShowVideo',
382
- 'setCaller',
383
- 'setMedicalOrPrescription',
384
- 'setModalVisible',
385
- 'setFooterMessage',
386
- 'setQueueItem',
387
- 'setSessionId',
388
- 'setCurrentTab',
389
- 'setClientId',
390
- 'setOnChating',
391
- 'setClientParams',
392
- 'setSessionHistoryList',
393
- 'setSessionEnd',
394
- 'setQuestionId',
395
- 'setDictionaryKey',
396
- 'setMultiVideoShow',
397
- 'setVideoStatus',
398
- 'setVideoMode',
399
- 'setAddMemberType',
400
- 'setShowAudio',
401
- 'setVideoMembers',
402
- 'setLastCurrentTab',
403
- 'setIsRecorderVoice',
404
- 'setVideoData',
405
- 'setSessionType',
406
- 'setCallerStatus'
407
- ]),
408
- ...mapMutations({
409
- clearMsgList: 'setMsgList'
410
- }),
411
- ...mapActions(['sendMessage', 'sendRobotMessage', 'handleBotChat', 'setMsgList', 'setChatTimer']),
412
- deleteStyle(value) {
413
- // 去掉样式
414
- value = value.replace(/style\s*?=\s*?(['"])[\s\S]*?\1/g, '');
415
- value = value.replace(/data\s*?=\s*?(['"])[\s\S]*?\1/g, '');
416
- value = value.replace(/class\s*?=\s*?(['"])[\s\S]*?\1/g, match => {
417
- if (match.includes('twa')) {
418
- return match;
419
- } else {
420
- return '';
421
- }
422
- });
423
- value = value.replace(/face\s*?=\s*?(['"])[\s\S]*?\1/g, '');
424
- value = value.replace(/<a/g, '<span');
425
- value = value.replace(/a>/g, 'span>');
426
- value = value.replace(/<i/g, '<span');
427
- value = value.replace(/i>/g, 'span>');
428
- value = value.replace(/<u/g, '<span');
429
- value = value.replace(/u>/g, 'span>');
430
- value = value.replace(/<strong/g, '<span');
431
- value = value.replace(/strong>/g, 'span>');
432
- value = value.replace(/<h\d/g, '<span');
433
- value = value.replace(/h\d>/g, 'span>');
434
- value = value.replace(/<b>/g, '<span>');
435
- value = value.replace(/b>/g, 'span>');
436
- value = value.replace(/<font/g, '<span');
437
- value = value.replace(/font>/g, 'span>');
438
- return value;
439
- },
440
- // 更新 message值 (未用v-html,修改后需要手动设置聊天框html内容)
441
- updateMessage(value) {
442
- this._editValue = value;
443
- this.setFooterMessage(`${value}`);
444
- const editRef = this.$refs.contentEdit;
445
- if (!editRef) return;
446
- editRef.innerHTML = value;
447
- },
448
- // 根据 聊天框 html节点内容 设置 message值 (未用v-html,html内容更改需主动设置message值)
449
- setMessageByHTML() {
450
- const editRef = this.$refs.contentEdit;
451
- if (!editRef) return;
452
- const message = editRef.innerHTML;
453
- this._editValue = message;
454
- this.setFooterMessage(`${message}`);
455
- },
456
-
457
- showEmoji() {
458
- // this.visible = true;
459
- },
460
-
461
- isShow(type) {
462
- if (type === 'SMILINEFACE' && this.curChatType === 'robot') return true;
463
- if (this.isListClassify) {
464
- let label;
465
- if (this.isServer) {
466
- label = 'serverFunctionSetting';
467
- } else {
468
- label = 'clientFunctionSetting';
469
- }
470
- let classifyItem = this.assemblySetting.listClassify.filter(item => item.classify === this.classify);
471
- if (!classifyItem || !classifyItem.length) return;
472
- let funs = classifyItem[0][label] || [];
473
- return funs.find(item => item.function === type)?.isChecked === 'Y';
474
- }
475
- let label;
476
- if (this.isServer) {
477
- label = 'serverSetting';
478
- } else {
479
- label = 'clientSetting';
480
- }
481
- let funs = this.assemblySetting[label]?.functionSetting || [];
482
- return funs.find(item => item.function === type)?.isChecked === 'Y';
483
- },
484
-
485
- handleEmojiClick(item) {
486
- let emojiHtml = `<p contenteditable="false" class="twa emoji ${item}"></p>&nbsp`;
487
- this.insertToEdit(emojiHtml);
488
- },
489
- /* edit-content光标定位 */
490
- resetSelection() {
491
- let edit = this.$refs.contentEdit;
492
- let sel, range;
493
- if (window.getSelection && document.createRange) {
494
- range = document.createRange();
495
- range.selectNodeContents(edit);
496
- range.collapse(true);
497
- range.setEnd(edit, edit.childNodes.length);
498
- range.setStart(edit, edit.childNodes.length);
499
- sel = window.getSelection();
500
- sel.removeAllRanges();
501
- sel.addRange(range);
502
- }
503
- },
504
-
505
- insertToEdit(value, isEditing) {
506
- const { _editSel, _editRange } = this;
507
- let sel, range;
508
- if (window.getSelection) {
509
- if (!isEditing) {
510
- if (_editRange) {
511
- sel = _editSel;
512
- range = _editRange;
513
- } else {
514
- // 初始状态没有focus聊天框,需要手动focus 获取光标位置
515
- const editRef = this.$refs.contentEdit;
516
- if (!editRef) return;
517
- editRef.focus();
518
- this.saveRange();
519
- sel = this._editSel;
520
- range = this._editRange;
521
- }
522
- } else {
523
- sel = window.getSelection();
524
- range = sel.getRangeAt(0);
525
- }
526
- range.deleteContents();
527
- let frag;
528
- if (range.createContextualFragment) {
529
- frag = range.createContextualFragment(value);
530
- } else {
531
- const el = document.createElement('div');
532
- el.innerHTML = value;
533
- frag = document.createDocumentFragment();
534
- let node, lastNode;
535
- while ((node = el.firstChild)) {
536
- lastNode = frag.appendChild(node); // 之所以能终止循环,是因为el的child一旦添加到dom后,作为fragment里的的child就没了。那么el.firstChild就会返回null
537
- }
538
- }
539
- const lastNode = frag.lastChild;
540
- range.insertNode(frag);
541
- // 把光标挪到插入的元素后面
542
- range.setStartAfter(lastNode);
543
- sel.removeAllRanges();
544
- sel.addRange(range);
545
- } else {
546
- range = isEditing || !_editRange ? document.selection.createRange() : _editRange;
547
- range.pasteHTML(value);
548
- range.select();
549
- }
550
- this.setMessageByHTML();
551
- },
552
- saveRange() {
553
- if (window.getSelection) {
554
- this._editSel = window.getSelection();
555
- this._editRange = this._editSel.getRangeAt(0);
556
- } else {
557
- this._editRange = document.selection.createRange();
558
- }
559
- },
560
- handleInput(e) {
561
- if (e.inputType === 'deleteContentBackward') {
562
- this.screenshot = false;
563
- this.screenshotFile = null;
564
- }
565
- setTimeout(() => {
566
- if (this.ping || this.pasteTimer) return;
567
- this.setMessageByHTML();
568
- }, 0);
569
- },
570
- changeData(e) {
571
- let message = e.srcElement.innerHTML;
572
- if (e.inputType === 'deleteContentBackward') {
573
- let div = document.createElement('div');
574
- div.innerHTML = e.srcElement.innerHTML;
575
- if (div.lastChild?.className?.startsWith('twa emoji')) {
576
- div.removeChild(div.lastChild);
577
- message = div.innerHTML;
578
- }
579
- this.screenshot = false;
580
- this.screenshotFile = null;
581
- }
582
-
583
- setTimeout(() => {
584
- if (this.ping) return;
585
- this.setFooterMessage(`${message}`);
586
- this.$nextTick().then(() => {
587
- this.resetSelection();
588
- this.ping = false;
589
- });
590
- }, 0);
591
- },
592
-
593
- beforeUpload(data) {
594
- let { type } = data;
595
- if (!type.startsWith('image/')) {
596
- this.$message.warning(`${this.i18nText('1.9.357')}!`);
597
- return;
598
- }
599
- let that = this;
600
- this.$confirm({
601
- title: this.i18nText('1.9.331'),
602
- content: `${this.i18nText('1.9.358')}?`,
603
- okText: '确定',
604
- cancelText: '取消',
605
- onOk() {
606
- return that.uploadImg(data);
607
- },
608
- onCancel() {
609
- return false;
610
- }
611
- });
612
- },
613
-
614
- /* 压缩上传 */
615
- uploadImg(file) {
616
- const hide = this.$message.loading(`${this.i18nText('1.9.199')}...`, 0);
617
- new ImageCompress(file, 'canvas', fetch, {
618
- url: '/hospitalUpload/picture'
619
- })
620
- .execute()
621
- .then(({ data }) => {
622
- if (data.result === 'SUCCESS') {
623
- this.doMsgSend(1, data.map.url);
624
- hide();
625
- } else {
626
- hide();
627
- this.$message.info(data.msg || `${this.i18nText('1.9.174')}!`);
628
- }
629
- })
630
- .catch(() => {
631
- hide();
632
- this.$message.info(`${this.i18nText('1.9.174')}!`);
633
- })
634
- .finally(() => {
635
- this.screenshotFile = null;
636
- this.screenshot = false;
637
- this.updateMessage('');
638
- });
639
- },
640
-
641
- /* 发送消息 */
642
- doMsgSend(type, content) {
643
- let body = {
644
- content: type === 0 ? `<div>${content}</div>` : content,
645
- type: type
646
- };
647
- if (this.curChatType === 'robot') {
648
- // this.setQuestionId("");
649
- // this.setDictionaryKey("");
650
- this.sendRobotMessage(content);
651
- } else {
652
- this.sendMessage(body);
653
- }
654
- },
655
- handleMsgSend() {
656
- // 截图上传
657
- if (this.screenshot) {
658
- this.uploadImg(this.screenshotFile);
659
- return;
660
- }
661
- if (this.chatType === 'bot') {
662
- this.handleBotChat(this.message).then(isHuman => {
663
- if (isHuman) {
664
- this.$parent.$parent.$parent.initWebSocket();
665
- }
666
- this.updateMessage('');
667
- });
668
- return;
669
- }
670
-
671
- const editRef = this.$refs.contentEdit;
672
- if (editRef) {
673
- const message = editRef.innerText;
674
- if (message.length > 2000) {
675
- this.$message.warning('请控制在2000字以内!');
676
- return;
677
- }
678
- }
679
- // let reg = /(.*?)(<div><br><\/div>)*?$/;
680
- let reg = /(.*?)(<div>(<span>)?<br>(<\/span>)?<\/div>)*?$/;
681
- let message = this.message.match(reg);
682
- if (message[2]) {
683
- this.updateMessage(message[1]);
684
- }
685
- this.updateMessage(this.message.replace(/style/g, 'data'));
686
- if (this.message.trim()) {
687
- this.doMsgSend(0, this.message);
688
- } else {
689
- return this.$message.warning(this.i18nText('1.9.359'));
690
- }
691
- this.updateMessage('');
692
- },
693
- handleKeyDown(e) {
694
- // 判断enter 发送还是换行
695
- if (e.keyCode === 13) {
696
- e.preventDefault();
697
- if (e.altKey || e.ctrlKey || e.metaKey || e.shiftKey) {
698
- this.handleWrap();
699
- } else {
700
- this.handleMsgSend();
701
- }
702
- }
703
- },
704
- handleWrap() {
705
- let ctn = `<br>`;
706
- // 非ie环境 元素后面没有node的时候换行需要两个<br>
707
- if (!/msie/.test(navigator.userAgent) && window.getSelection) {
708
- const focusNode = window.getSelection().focusNode;
709
- let nextSibling = focusNode.nextSibling;
710
- if (focusNode === this.$refs.contentEdit) {
711
- nextSibling = focusNode.childNodes[0];
712
- }
713
- do {
714
- if (!nextSibling || nextSibling.nodeValue || nextSibling.tagName == 'BR') break;
715
- } while ((nextSibling = nextSibling.nextSibling));
716
- if (!nextSibling) {
717
- ctn += ctn;
718
- }
719
- }
720
- this.insertToEdit(ctn);
721
- this.$nextTick().then(() => {
722
- this.ping = false;
723
- });
724
- },
725
- handleVideoClick(type) {
726
- this.setCallerStatus('')
727
- // this.curScrollItem.type 2多人 1单人
728
- if (this.sessionType === 2) {
729
- this.$emit('update:addvisible', true);
730
- this.setAddMemberType(2);
731
- this.setVideoMode(type === 'voice' ? 1 : 2);
732
- } else {
733
- const openTalking = sip => {
734
- this.setShowAudio(true);
735
- this.setVideoMode(type === 'voice' ? 1 : 2);
736
- this.$nextTick(() => {
737
- this.setVideoMembers([]);
738
- });
739
- this.setVideoData({
740
- callerSipKey: sip?.username,
741
- callerSipName: sip?.name
742
- });
743
- };
744
- if (this.validTalkingEnv) {
745
- this.validTalkingEnv().then(res => {
746
- if (res.status) {
747
- openTalking(res.sip);
748
- }
749
- });
750
- } else {
751
- openTalking();
752
- }
753
- }
754
- // this.groupMembers.length > 2
755
- return;
756
- },
757
- handleVoice(val) {
758
- val && this.setIsRecorderVoice(true);
759
- this.showVoice = val;
760
- },
761
- showPrescription() {
762
- this.setModalVisible(true);
763
- this.setMedicalOrPrescription('prescription');
764
- },
765
- showMedicalRecord() {
766
- this.setModalVisible(true);
767
- this.setMedicalOrPrescription('medical');
768
- },
769
- beofreToolBarClick(item) {
770
- // 操作栏前置条件
771
- return new Promise((resolve, reject) => {
772
- const { precondition } = item;
773
- if (precondition) {
774
- this.requestPreCondition(precondition)
775
- .then(res => {
776
- const result = res.data.result === 'SUCCESS';
777
- !result && this.$message.error(res.data.resultMsg || this.i18nText('1.9.22'));
778
- resolve(result);
779
- })
780
- .catch(error => {
781
- this.$message.error(error.resultMsg || this.i18nText('1.9.22'));
782
- resolve(false);
783
- });
784
- } else {
785
- resolve(true);
786
- }
787
- });
788
- },
789
- async handleToolBarClick(item) {
790
- const validate = await this.beofreToolBarClick(item);
791
- if (!validate) return;
792
- let { openMode, name, templateId, isFullScreen } = item;
793
- this.dispatchEvent('click_toolbar', { ...item });
794
- if (!item.address) return;
795
- this.curToolbarItem = item;
796
- this.templateId = templateId;
797
- this.isFullScreen = isFullScreen === 'Y';
798
- let address = this.getLinkAddress(item);
799
- if (openMode === 'WINDOW') {
800
- window.open(address, name);
801
- return;
802
- }
803
- if (openMode === 'EJECT') {
804
- this.modalShow = true;
805
- this.modalData = Object.assign({}, item, { address });
806
- this.$nextTick().then(() => {
807
- window.addEventListener('message', this.iframeEvent);
808
- });
809
- return;
810
- }
811
- },
812
- iframeEvent(event) {
813
- const method = event?.data?.method;
814
- this[method]?.(event);
815
- },
816
- // 发送
817
- handleParentMessageSend({ data, source, origin }) {
818
- let params = {
819
- assemblyId: this.assemblyId,
820
- sessionId: this.sessionId,
821
- orgId: this.orgId
822
- };
823
- params = Object.assign(params, data.data);
824
- let { actionType } = this.curToolbarItem;
825
- actionType && Object.assign(params, { actionType });
826
- return fetch.post('/chat/access/sendToolBarData', qs.stringify(params)).then(({ data }) => {
827
- if (data.result === 'SUCCESS') {
828
- source.postMessage({ status: 0, resultMsg: data.resultMsg }, origin);
829
- } else {
830
- source.postMessage({ status: 1, resultMsg: data.resultMsg }, origin);
831
- }
832
- });
833
- },
834
- // 关闭
835
- handleClose() {
836
- this.modalShow = false;
837
- window.removeEventListener('message', this.iframeEvent);
838
- },
839
- getLinkAddress(item,paramsData= {}) {
840
- let t = {talbe:{},form:{},sys:{}};
841
- Object.assign(t,paramsData ||{})
842
- let { address = '', params = [] } = item;
843
- let urlParams = [];
844
- params.forEach(({ p_name, p_value }) => {
845
- let value;
846
- if (p_value.startsWith('form.')) {
847
- value = this.clientParams[p_value.slice(5)];
848
- } else if( /\${result.(.*?)\}/g.test(p_value)) {
849
- value = vexutils.handleSysParams(p_value,t)
850
- } else {
851
- value = p_value;
852
- }
853
- urlParams.push(`${p_name}=${value}`);
854
- });
855
- if(/\${result.(.*?)\}/g.test(address)){
856
- address = vexutils.handleSysParams(address,t)
857
- }
858
- let { actionType } = this.curToolbarItem;
859
- actionType && urlParams.push(`actionType=${actionType}`);
860
- if (address.includes('?')) {
861
- address += `&${urlParams.join('&')}`;
862
- } else {
863
- address += `?${urlParams.join('&')}`;
864
- }
865
- return address;
866
- },
867
- handleOk() {
868
- window.handleParentMessageSend();
869
- },
870
- // 剪切板粘贴
871
- handlePaste(e) {
872
- const cbd = e.clipboardData;
873
- const ua = window.navigator.userAgent;
874
- // 如果是 Safari 直接 return
875
- if (!(e.clipboardData && e.clipboardData.items)) {
876
- this.handlePasteText(e);
877
- return;
878
- }
879
- if (
880
- cbd.items &&
881
- cbd.items.length === 2 &&
882
- cbd.items[0].kind === 'string' &&
883
- cbd.items[1].kind === 'file' &&
884
- cbd.types &&
885
- cbd.types.length === 2 &&
886
- cbd.types[0] === 'text/plain' &&
887
- cbd.types[1] === 'Files' &&
888
- ua.match(/Macintosh/i) &&
889
- Number(ua.match(/Chrome\/(\d{2})/i)[1]) < 49
890
- ) {
891
- this.handlePasteText(e);
892
- return;
893
- }
894
- const hasFile = this.handleImgPaste(cbd.items);
895
- if (hasFile) {
896
- e.stopPropagation();
897
- e.preventDefault();
898
- return;
899
- }
900
- this.handlePasteText(e);
901
- },
902
- handleImgPaste(files) {
903
- let hasFile = false;
904
- for (let i = 0; i < files.length; i++) {
905
- let item = files[i];
906
- if (item.kind == 'file') {
907
- hasFile = true;
908
- // blob 就是从剪切板获得的文件,可以进行上传或其他操作
909
- const blob = item.getAsFile();
910
- if (blob.size === 0) {
911
- return;
912
- }
913
- const reader = new FileReader();
914
- const imgs = new Image();
915
- imgs.file = blob;
916
- reader.onload = e => {
917
- imgs.src = e.target.result;
918
- imgs.width = '150';
919
- this.updateMessage(this.htmlToString(imgs));
920
- this.$nextTick().then(() => {
921
- this.resetSelection();
922
- this.screenshot = true;
923
- this.screenshotFile = blob;
924
- });
925
- };
926
- reader.readAsDataURL(blob);
927
- }
928
- }
929
- return hasFile;
930
- },
931
- // 处理文本粘贴
932
- handlePasteText() {
933
- if (this.pasteTimer) clearTimeout(this.pasteTimer);
934
- const ele = this.$refs.contentEdit;
935
- if (!ele) return;
936
- let before = ele.innerHTML;
937
- this.pasteTimer = setTimeout(() => {
938
- function getPasteHtml(before, after) {
939
- // 计算粘贴后html不同的位置起点和终点(插入"abc",则得出sPos为0,ePos为2)
940
- var sPos = -1,
941
- ePos = -1;
942
- for (var i = 0, len = after.length; i < len; ++i) {
943
- if (sPos == -1 && before.substr(i, 1) != after.substr(i, 1)) sPos = i;
944
- if (ePos == -1 && before.substr(before.length - i - 1, 1) != after.substr(after.length - i - 1, 1)) ePos = i;
945
-
946
- if ((sPos != -1 && ePos != -1) || len - 1 - ePos <= sPos) break;
947
- }
948
- if (sPos == -1 || ePos == -1) return;
949
- ePos = len - 1 - ePos;
950
-
951
- if (ePos <= sPos) {
952
- // 遇到像ab,粘贴ab后变成abab这种了,这是需要从sPos开始往后找,找到和before一样的为止
953
- i = sPos;
954
- var beforeSPosNext = before.substr(i + 1, 10); // 取后10个来对比,如果没有10个,则取最多
955
- while (++i < len) {
956
- if (beforeSPosNext == after.substr(i, beforeSPosNext.length)) {
957
- ePos = i;
958
- break;
959
- }
960
- }
961
- i == len && (ePos = len - 1);
962
- }
963
- // 如果有<和>,则需要计算在内
964
- if (after.substr(sPos - 1, 1) == '<') --sPos;
965
- if (after.substr(ePos + 1, 1) == '>') ++ePos;
966
-
967
- // 分段获得文本,前半段 插入段 后半段
968
- var pastedText = after.substring(sPos, ePos + 1),
969
- formerText = after.substr(0, sPos),
970
- latterText = after.substr(sPos + pastedText.length);
971
-
972
- // 判断formerText和pastedText是否闭合了
973
- var lastLT = formerText.lastIndexOf('<'),
974
- lastGT = formerText.lastIndexOf('>');
975
- if (lastGT < lastLT) {
976
- // 标签断了xxxxx<
977
- pastedText = formerText.slice(lastLT) + pastedText;
978
- formerText = formerText.slice(0, lastLT);
979
- }
980
-
981
- lastLT = pastedText.lastIndexOf('<');
982
- lastGT = pastedText.lastIndexOf('>');
983
- if (lastGT < lastLT) {
984
- // 标签断了<xxxxxx
985
- var _insertGT = latterText.indexOf('>') + 1;
986
- pastedText += latterText.slice(0, _insertGT);
987
- latterText = latterText.slice(_insertGT);
988
- }
989
- return [formerText, pastedText, latterText];
990
- }
991
- var after = ele.innerHTML;
992
- const data = getPasteHtml(before, after);
993
- if (data) {
994
- let [formerText, insertText, latterText] = data;
995
- insertText = this.deleteStyle(insertText);
996
- ele.innerHTML = formerText + insertText + "<span class='pasteCaretPosHelper'></span>" + latterText; // 插入pasteCaretPosHelper以帮助定位光标位置
997
- var pasteCaretPosHelper = ele.querySelector('.pasteCaretPosHelper'),
998
- range,
999
- selection;
1000
- if (pasteCaretPosHelper) {
1001
- if (document.createRange) {
1002
- // Chrome, IE 9+
1003
- range = document.createRange();
1004
- range.setStartAfter(pasteCaretPosHelper);
1005
- range.collapse(false);
1006
- selection = window.getSelection();
1007
- selection.removeAllRanges();
1008
- selection.addRange(range);
1009
- } else if (document.selection) {
1010
- // IE 8 and lower
1011
- range = document.body.createTextRange();
1012
- range.moveToElementText(pasteCaretPosHelper);
1013
- range.collapse(false);
1014
- range.select();
1015
- }
1016
- pasteCaretPosHelper.parentNode.removeChild(pasteCaretPosHelper);
1017
- }
1018
- }
1019
- this.setMessageByHTML();
1020
- this.saveRange();
1021
- this.pasteTimer = null;
1022
- }, 50);
1023
- },
1024
- // 拖拽上传图片
1025
- handleFiles(files) {
1026
- this.handleImgPaste(files);
1027
- },
1028
- htmlToString(dom) {
1029
- let div = document.createElement('div');
1030
- div.appendChild(dom);
1031
- return div.innerHTML;
1032
- },
1033
- /**
1034
- * 接入拦截
1035
- */
1036
- handleAccessIntercept(result) {
1037
- let { data } = result;
1038
- let { status, data: param } = data;
1039
- if (status === 1) {
1040
- this.accessChat(param).finally(() => {
1041
- this.accessIntercept = false;
1042
- });
1043
- } else {
1044
- this.accessIntercept = false;
1045
- }
1046
- },
1047
- /* 客服点击接入按钮进入会话 */
1048
- async handleAccess() {
1049
- this.apiResult = {};
1050
- let accessInterceptValidate = this.handleAccessInterceptValue();
1051
- let data = null; // 接口参数
1052
- if (accessInterceptValidate && this.accessInterceptUrl) {
1053
- const r = await this.beforeAccessIntercept();
1054
- if (!r) return;
1055
- if (r.value == 'Y') {
1056
- this.accessIntercept = true;
1057
- this.$nextTick().then(() => {
1058
- window.addEventListener('message', this.handleAccessIntercept);
1059
- });
1060
- return;
1061
- } else {
1062
- data = r.data; // 获取接入拦截返回参数
1063
- }
1064
- }
1065
- clearTimeout(this.timer);
1066
- this.timer = setTimeout(() => {
1067
- this.accessChat(data);
1068
- }, 300);
1069
- },
1070
- // 条件转换请求
1071
- requestPreCondition(precondition) {
1072
- const { targetType, targetName, targetId, params } = precondition;
1073
- const data = parseParams(params, this.clientParams);
1074
- let url = '';
1075
- if (targetType === 'API') {
1076
- data.url = targetName;
1077
- url = `/api/executeApi`;
1078
- } else {
1079
- data.sqlEnterId = targetId;
1080
- url = `/form/saveSqlEnter`;
1081
- }
1082
- return fetch.post(url, qs.stringify(data));
1083
- },
1084
- async beforeAccessIntercept() {
1085
- function parseResult(r) {
1086
- let value = 'Y';
1087
- const data = r.split('&').reduce((obj, str) => {
1088
- const [k, v] = str.split('=');
1089
- if (k) {
1090
- if (k == 'value') {
1091
- value = v || value;
1092
- } else {
1093
- obj = obj || {};
1094
- obj[k] = v || '';
1095
- }
1096
- }
1097
- return obj;
1098
- }, null);
1099
- return {
1100
- value,
1101
- data
1102
- };
1103
- }
1104
- this.apiResult = {};
1105
- const accessInterceptSetting = this.assemblySetting.accessInterceptSetting || [];
1106
- const setting = accessInterceptSetting.find(item => item.source == 'PC' && item.isChecked == 'Y' && item.precondition);
1107
- if (setting) {
1108
- const precondition = setting.precondition;
1109
- try {
1110
- let res = await this.requestPreCondition(precondition);
1111
- this.apiResult = res?.data?.map || {};
1112
- if (res.data.result === 'SUCCESS') {
1113
- let value = res.data.map?.result;
1114
- value = value ? JSON.parse(value) : {};
1115
- return value.value ? parseResult(value.value) : false;
1116
- } else {
1117
- this.$message.error(res.data.resultMsg || this.i18nText('1.9.22'));
1118
- return false;
1119
- }
1120
- } catch (error) {
1121
- this.$message.error(error.resultMsg || this.i18nText('1.9.22'));
1122
- return false;
1123
- }
1124
- }
1125
- return {
1126
- value: 'Y',
1127
- data: null
1128
- };
1129
- },
1130
- handleAccessInterceptValue() {
1131
- let { listClassify = [], isListClassify, serverSetting = {} } = this.assemblySetting;
1132
- let list = [];
1133
- if (isListClassify != 'Y') {
1134
- let { functionSetting = [] } = serverSetting;
1135
- list = functionSetting;
1136
- } else {
1137
- let curList = listClassify.find(item => item.classify == this.classify);
1138
- if (!curList) return false;
1139
- list = curList.serverFunctionSetting || [];
1140
- }
1141
- if (!list.length) return true;
1142
- let curItem = list.find(item => item.function == 'ACCESS_INTERCEPT');
1143
- if (!curItem) return true;
1144
- return curItem.isChecked == 'Y';
1145
- },
1146
- accessChat(data) {
1147
- let { bid: userId, username, userType } = this.queueItem;
1148
- let params = {
1149
- assemblyId: this.assemblyId,
1150
- userId,
1151
- userType,
1152
- lastSessionId: this.sessionId,
1153
- relateId: this.clientParams.relateId
1154
- };
1155
- if (data) {
1156
- params.addParams = JSON.stringify(data);
1157
- }
1158
- return fetch.post('/chat/service/accessChat', qs.stringify(params)).then(({ data }) => {
1159
- if (data.result === 'SUCCESS') {
1160
- let { countdown, clientParams, sessionType, countdownDesc } = data.map;
1161
- this.setSessionId(data.map.sessionId);
1162
- this.clearMsgList();
1163
- this.setMsgList();
1164
- this.setCurrentTab('session');
1165
- this.setLastCurrentTab('session');
1166
- this.setClientId(username);
1167
- this.setOnChating(true);
1168
- this.setSessionEnd(false);
1169
- this.setQueueItem(null);
1170
- sessionType && this.setSessionType(sessionType);
1171
- if (countdown >= 0) {
1172
- this.setChatTimer({ countdown, countdownDesc });
1173
- }
1174
- this.setClientParams(clientParams);
1175
- this.getSessionHistoryList();
1176
- this.dispatchEvent('user_accessSuccess', {
1177
- map: data.map,
1178
- user: { ...this.curScrollItem }
1179
- });
1180
- } else {
1181
- this.$message.warning(data.resultMsg);
1182
- }
1183
- });
1184
- },
1185
- handleOpenTransfer() {
1186
- if (this.enableTransfer) {
1187
- this.transferVisible = true;
1188
- }
1189
- },
1190
- handleTransfer(data) {
1191
- const receiveId = data.converUserId;
1192
- if (receiveId) {
1193
- if (this.userId == receiveId) {
1194
- this.$message.warning(this.i18nText('1.2.7.14.1'));
1195
- return;
1196
- }
1197
- this.transferVisible = false;
1198
- fetch({
1199
- url: '/chat/service/transferQueue',
1200
- method: 'POST',
1201
- data: qs.stringify({
1202
- relateId: this.clientParams.relateId,
1203
- orgId: this.orgId,
1204
- assemblyId: this.assemblyId,
1205
- userId: this.curScrollItem?.bid,
1206
- userType: this.curScrollItem?.userType || '',
1207
- receiveId: receiveId,
1208
- deptId: data.deptId,
1209
- remark: data.converMark
1210
- })
1211
- })
1212
- .then(({ data }) => {
1213
- if (data.result === 'SUCCESS') {
1214
- this.dispatchEvent('user_transferQueueSuccess', {
1215
- user: { ...this.curScrollItem }
1216
- });
1217
- this.$message.success(this.i18nText('1.9.336'));
1218
- this.clearCurrentSession(true);
1219
- } else {
1220
- this.$message.warning(data.resultMsg || this.i18nText('1.10.201') + this.i18nText('1.9.22'));
1221
- }
1222
- })
1223
- .catch(err => {
1224
- this.$message.warning(err.resultMsg || this.i18nText('1.10.201') + this.i18nText('1.9.22'));
1225
- });
1226
- } else {
1227
- this.$message.warning(this.i18nText('1.9.335'));
1228
- }
1229
- },
1230
- getSessionHistoryList() {
1231
- this.setSessionHistoryList([]);
1232
- if (this.assemblySetting.recordScopeSetting === 'CURRENT') return;
1233
- fetch
1234
- .get('/chat/service/getPastSessionList', {
1235
- params: { sessionId: this.sessionId }
1236
- })
1237
- .then(({ data }) => {
1238
- if (data.result === 'SUCCESS') {
1239
- let list = data.list.map(item => {
1240
- if (item.id === this.sessionId || this.type === 'session') {
1241
- item.onChating = true;
1242
- }
1243
- return item;
1244
- });
1245
- this.setSessionHistoryList(list);
1246
- }
1247
- });
1248
- },
1249
- openQuickData() {
1250
- // this.quickShow = true;
1251
- if (this.quickData.length < 1) {
1252
- fetch
1253
- .get('/fastReply/getChatList', {
1254
- params: {
1255
- category: this.assemblySetting.chatPublicCategory
1256
- }
1257
- })
1258
- .then(({ data }) => {
1259
- if (data.result === 'SUCCESS') {
1260
- this.quickData = data.list;
1261
- }
1262
- });
1263
- }
1264
- },
1265
- // 清除当前会话
1266
- clearCurrentSession(update = false) {
1267
- this.clearMsgList([]);
1268
- this.setSessionHistoryList([]);
1269
- this.setSessionTimer('');
1270
- this.setClientId('');
1271
- this.setSessionId('');
1272
- this.setScrollQueueId('');
1273
- this.setLastClassify('');
1274
- this.setCurScrollItem(null);
1275
- this.setQueueItem(null);
1276
- this.setClientParams(null);
1277
- if (update) {
1278
- // 重新设置 currentTab 触发 scrollList 组件 更新排队列表
1279
- const currentTab = this.store.state?.currentTab;
1280
- if (currentTab === 'queue') {
1281
- this.setCurrentTab('');
1282
- this.$nextTick(() => {
1283
- this.setCurrentTab(this.store.state?.currentTab || currentTab);
1284
- });
1285
- }
1286
- }
1287
- },
1288
- openRefuseReason() {
1289
- if (this.enableRefuse) {
1290
- this.refuseVisible = true;
1291
- }
1292
- },
1293
- handleRefuse(remark) {
1294
- if (!this.enableRefuse) return;
1295
- fetch({
1296
- url: '/chat/service/accessRefuse',
1297
- method: 'POST',
1298
- data: qs.stringify({
1299
- relateId: this.clientParams.relateId,
1300
- orgId: this.orgId,
1301
- assemblyId: this.assemblyId,
1302
- userId: this.curScrollItem?.bid,
1303
- userType: this.curScrollItem?.userType || '',
1304
- remark: remark || '其他'
1305
- })
1306
- })
1307
- .then(({ data }) => {
1308
- if (data.result === 'SUCCESS') {
1309
- this.dispatchEvent('user_refuseSuccess', {
1310
- user: { ...this.curScrollItem }
1311
- });
1312
- this.$message.success(this.i18nText('1.10.184'));
1313
- this.clearCurrentSession(true);
1314
- } else {
1315
- this.$message.warning(data.resultMsg || this.i18nText('1.9.353') + this.i18nText('1.9.22'));
1316
- }
1317
- })
1318
- .catch(err => {
1319
- this.$message.warning(err.resultMsg || this.i18nText('1.9.353') + this.i18nText('1.9.22'));
1320
- });
1321
- this.refuseVisible = false;
1322
- },
1323
- bindEvents() {
1324
- const handleDrop = e => {
1325
- e.stopPropagation();
1326
- e.preventDefault();
1327
- this.handleFiles(e.dataTransfer.items);
1328
- };
1329
- document.addEventListener('drop', handleDrop, false);
1330
- this.registerEvent('accessCurrentSession', this.handleAccess);
1331
- this.registerEvent('refuseCurrentSession', this.openRefuseReason);
1332
- this.registerEvent('transferCurrentQueueSession', this.handleOpenTransfer);
1333
- this.$on('hook:beforeDestroy', () => {
1334
- document.removeEventListener('drop', handleDrop, false);
1335
- window.removeEventListener('message', this.iframeEvent);
1336
- window.removeEventListener('message', this.handleAccessIntercept);
1337
- this.unregisterEvent('accessCurrentSession', this.handleAccess);
1338
- this.unregisterEvent('refuseCurrentSession', this.openRefuseReason);
1339
- this.unregisterEvent('transferCurrentQueueSession', this.handleOpenTransfer);
1340
- });
1341
- }
1342
- },
1343
- watch: {
1344
- accessIntercept: {
1345
- handler(val) {
1346
- if (!val) {
1347
- window.removeEventListener('message', this.handleAccessIntercept);
1348
- }
1349
- }
1350
- },
1351
- message: {
1352
- handler(v) {
1353
- // 使用 _editValue 判断是否组件外修改 message,并更新聊天框
1354
- if (this._editValue !== v) {
1355
- this.updateMessage(v);
1356
- }
1357
- }
1358
- }
1359
- }
1360
- };
1361
- </script>
1362
-
1363
- <style lang="less">
1364
- .chat-footer-modal-big {
1365
- padding: 10px;
1366
- &.ant-modal-wrap {
1367
- overflow: none;
1368
- }
1369
- .ant-modal {
1370
- width: 100%;
1371
- height: 100%;
1372
- top: 0;
1373
- padding-bottom: 0;
1374
- }
1375
- .ant-modal-content {
1376
- height: 100%;
1377
- }
1378
- .ant-modal-body {
1379
- height: calc(100% - 55px);
1380
- }
1381
- }
1382
- .chat-footer-modal-small {
1383
- &.ant-modal-wrap {
1384
- overflow: none;
1385
- }
1386
- .ant-modal {
1387
- width: 80%;
1388
- height: 80%;
1389
- padding-bottom: 0;
1390
- top: 60px;
1391
- }
1392
- .ant-modal-content {
1393
- height: 100%;
1394
- }
1395
- .ant-modal-body {
1396
- height: 100%;
1397
- padding: 0;
1398
- }
1399
- }
1400
-
1401
- </style>
1402
- <style lang="less" scoped>
1403
- .message-operate {
1404
- display: flex;
1405
- align-items: center;
1406
- height: 40px;
1407
- line-height: 18px;
1408
- background: #fff;
1409
- overflow-x: auto;
1410
- overflow-y: hidden;
1411
- border-top: 1px solid #e6e6e6;
1412
- &::-webkit-scrollbar {
1413
- height: 5px;
1414
- }
1415
- > img {
1416
- // > .operate-icon {
1417
- margin-left: 10px;
1418
- }
1419
- /deep/ .ant-tabs .ant-tabs-top-content > .ant-tabs-tabpane {
1420
- height: 300px !important;
1421
- overflow: auto !important;
1422
- }
1423
- /deep/ .ant-upload-list {
1424
- display: none;
1425
- }
1426
- /deep/ .ant-upload {
1427
- height: 32px;
1428
- vertical-align: middle;
1429
- display: inline-block;
1430
- }
1431
- .no-click {
1432
- pointer-events: none;
1433
- opacity: 0.4;
1434
- }
1435
- .svg-icon {
1436
- color: rgba(0, 0, 0, 0.6);
1437
- }
1438
- }
1439
- .emoji-wrapper {
1440
- width: 500px;
1441
- height: 300px;
1442
- overflow: auto;
1443
- display: grid;
1444
- grid-template-columns: repeat(10, 1fr);
1445
- .emoji {
1446
- display: inline-block;
1447
- font-size: 20px;
1448
- margin: 10px;
1449
- text-align: center;
1450
- }
1451
- &::-webkit-scrollbar {
1452
- width: 5px;
1453
- }
1454
- }
1455
- .message {
1456
- position: relative;
1457
- height: 150px;
1458
- .message-send {
1459
- position: absolute;
1460
- bottom: 8px;
1461
- right: 14px;
1462
- height: 40px;
1463
- border-radius: 20px;
1464
- width: 86px;
1465
- font-weight: 600;
1466
- font-size: 18px;
1467
- color: #fff;
1468
- background-color: @primary-color;
1469
- border-color: @primary-color;
1470
- &:hover {
1471
- background-color: rgba(@primary-color, 0.7);
1472
- border-color: rgba(@primary-color, 0.7);
1473
- }
1474
- &[disabled]:hover,
1475
- &[disabled] {
1476
- background-color: #f5f5f5;
1477
- border-color: #d9d9d9;
1478
- color: #bebebe;
1479
- }
1480
- }
1481
- .prompt {
1482
- position: absolute;
1483
- bottom: 16px;
1484
- right: 120px;
1485
- color: #a6a6a6;
1486
- }
1487
- }
1488
- .infinite-container {
1489
- overflow-y: auto;
1490
- }
1491
- .message-wrapper {
1492
- position: relative;
1493
- .loading-icon {
1494
- position: absolute;
1495
- top: 0;
1496
- left: 50%;
1497
- }
1498
- }
1499
- .content-edit {
1500
- width: 100%;
1501
- height: 100%;
1502
- padding: 0 16px 56px;
1503
- word-break: break-word;
1504
- &::-webkit-scrollbar {
1505
- width: 5px;
1506
- }
1507
- cursor: text;
1508
- > div {
1509
- height: 100%;
1510
- outline: none;
1511
- overflow-y: auto;
1512
- overflow-x: hidden;
1513
- }
1514
- }
1515
- .overlay {
1516
- position: absolute;
1517
- width: 100%;
1518
- height: 100%;
1519
- top: 0;
1520
- cursor: not-allowed;
1521
- }
1522
- .queue-page {
1523
- display: flex;
1524
- flex-direction: column;
1525
- align-items: center;
1526
- justify-content: center;
1527
- position: absolute;
1528
- width: 100%;
1529
- height: 100%;
1530
- top: 0;
1531
- background: #fff;
1532
- z-index: 500;
1533
- border-top: 1px solid #e6e6e6;
1534
- .queue-tips {
1535
- font-size: 16px;
1536
- color: #272727;
1537
- line-height: 22px;
1538
- margin-bottom: 16px;
1539
- }
1540
- .queue-btns {
1541
- display: flex;
1542
- justify-content: center;
1543
- }
1544
- .toSession {
1545
- width: 80px;
1546
- height: 80px;
1547
- line-height: 80px;
1548
- text-align: center;
1549
- background: linear-gradient(180deg, #ffffff 0%, #f5f5f5 100%);
1550
- box-shadow: 0px 4px 9px 0px rgba(186, 186, 186, 0.79);
1551
- border-radius: 50%;
1552
- color: #4b4b4b;
1553
- font-size: 20px;
1554
- font-weight: 500;
1555
- font-family: PingFangSC-Medium, PingFang SC;
1556
- transition: transform 0.3s ease;
1557
- &.primary {
1558
- background: linear-gradient(180deg, #2d7aff 0%, #2c5df4 100%);
1559
- box-shadow: 0px 5px 8px 0px rgba(44, 101, 247, 0.39);
1560
- color: #fff;
1561
- }
1562
- &.danger {
1563
- background: linear-gradient(180deg, #ff4b4b 0%, #c80606 100%);
1564
- box-shadow: 0px 5px 8px 0px rgba(217, 26, 26, 0.35);
1565
- color: #fff;
1566
- }
1567
- &:hover {
1568
- transform: scale(1.2);
1569
- }
1570
- & + .toSession {
1571
- margin-left: 36px;
1572
- }
1573
- }
1574
- }
1575
- .toolbar {
1576
- // margin-left: 10px;
1577
- margin: 0 5px;
1578
- font-size: 20px;
1579
- outline: none;
1580
- padding: 6px;
1581
- box-sizing: content-box;
1582
- &:hover {
1583
- background-color: #ebebeb;
1584
- border-radius: 6px;
1585
- }
1586
- }
1587
- .closed-toolbar {
1588
- z-index: 100;
1589
- }
1590
- /deep/ p {
1591
- margin-bottom: 0;
1592
- }
1593
- .click-voice {
1594
- color: #2d7aff;
1595
- }
1596
- </style>
1
+ <template>
2
+ <a-layout-footer style="position: relative;cursor:pointer" v-show="!queueItem || !hideQueueBtns">
3
+ <div class="message-operate">
4
+ <a-upload
5
+ v-if="isShow('PICTURE')"
6
+ class="operate-icon"
7
+ v-decorator="[
8
+ 'upload',
9
+ {
10
+ valuePropName: 'fileList'
11
+ }
12
+ ]"
13
+ :beforeUpload="beforeUpload"
14
+ >
15
+ <a-tooltip placement="top">
16
+ <template slot="title">{{ i18nText('1.9.181') }}</template>
17
+ <svg-icon icon-class="liaotiantupian1" class="toolbar"></svg-icon>
18
+ </a-tooltip>
19
+ </a-upload>
20
+ <a-popover :title="i18nText('1.9.354')" trigger="click" v-model="visible">
21
+ <a slot="content">
22
+ <div class="emoji-wrapper">
23
+ <p v-for="(item, index) in emojiList" :key="index" @click="() => handleEmojiClick(item)" :class="['twa', 'emoji', item]"></p>
24
+ </div>
25
+ </a>
26
+
27
+ <a-tooltip placement="top">
28
+ <template slot="title">{{ i18nText('1.9.354') }}</template>
29
+ <svg-icon icon-class="liaotianbiaoqing1" @click="showEmoji" class="toolbar" v-if="isShow('SMILINEFACE')"></svg-icon>
30
+ </a-tooltip>
31
+ </a-popover>
32
+ <template v-for="(item, index) in toolbar">
33
+ <a-tooltip placement="top" :key="index">
34
+ <template slot="title">{{ item.name }}</template>
35
+ <svg-icon :icon-class="item.icon" class="toolbar" :class="{ 'closed-toolbar': closedSession }" @click="handleToolBarClick(item)"></svg-icon>
36
+ </a-tooltip>
37
+ </template>
38
+ <!-- 快捷回复 -->
39
+ <a-tooltip placement="top" :title="'快捷回复'">
40
+ <a-popover trigger="click" overlayClassName="quick-reply" placement="topLeft" v-model="quickShow">
41
+ <template slot="content">
42
+ <quick-reply :quickShow.sync="quickShow" :quickData="quickData"></quick-reply>
43
+ </template>
44
+ <svg-icon icon-class="liaotiankuaijiehuifu1" class="toolbar" @click="openQuickData" v-if="isShow('FASTREPLY') && isServer"></svg-icon>
45
+ </a-popover>
46
+ </a-tooltip>
47
+ <a-tooltip :title="'语音通话'">
48
+ <svg-icon icon-class="liaotianyuyindianhua" @click="handleVideoClick('voice')" class="toolbar" v-if="isShow('VOICE_CALL')"></svg-icon>
49
+ <!-- <img
50
+ width="20"
51
+ src="../img/multi-voice.png"
52
+ @click="handleVideoClick('voice')"
53
+ v-if="isShow('VOICE_CALL') && isServer"
54
+ /> -->
55
+ </a-tooltip>
56
+ <a-tooltip :title="'发送语音'">
57
+ <svg-icon icon-class="liaotianyuyin1" @click="handleVoice(true)" class="toolbar" :class="{ 'click-voice': showVoice }" v-if="isShow('VOICE')"></svg-icon>
58
+ <!-- <img
59
+ width="20"
60
+ src="../img/multi-video.png"
61
+ @click="handleVideoClick('video')"
62
+ v-if="isShow('VIDEO') && isServer"
63
+ /> -->
64
+ </a-tooltip>
65
+ <a-tooltip :title="'视频通话'">
66
+ <svg-icon icon-class="liaotianshipin1" @click="handleVideoClick('video')" class="toolbar" v-if="isShow('VIDEO')"></svg-icon>
67
+ <!-- <img
68
+ width="20"
69
+ src="../img/multi-video.png"
70
+ @click="handleVideoClick('video')"
71
+ v-if="isShow('VIDEO') && isServer"
72
+ /> -->
73
+ </a-tooltip>
74
+ </div>
75
+ <div class="message" style="text-align: left">
76
+ <div class="content-edit">
77
+ <div
78
+ ref="contentEdit"
79
+ @click="saveRange"
80
+ @keyup.exact="saveRange"
81
+ @change="changeData"
82
+ @input="handleInput"
83
+ @compositionstart="() => (this.ping = true)"
84
+ @compositionend="() => (this.ping = false)"
85
+ @keydown="handleKeyDown"
86
+ @paste="handlePaste"
87
+ contenteditable="true"
88
+ ></div>
89
+ </div>
90
+ <span class="prompt">Enter {{ i18nText('1.2.1.11.6') }}, Ctrl + Enter {{ i18nText('1.9.356') }}</span>
91
+ <a-button :disabled="sendDisabled" class="message-send" @click="handleMsgSend">{{ i18nText('1.2.1.11.6') }}</a-button>
92
+ </div>
93
+ <div id="canvas-wrapper" style="display: none">
94
+ <canvas id="canvas"></canvas>
95
+ </div>
96
+ <template v-if="showOverlay">
97
+ <div class="overlay" v-if="(!onChating || sessionEnd) && !queueItem"></div>
98
+ </template>
99
+ <div class="queue-page" v-if="queueItem">
100
+ <div v-if="accessDesc" class="queue-tips">{{ accessDesc }}</div>
101
+ <div class="queue-btns">
102
+ <div v-if="enableTransfer" class="toSession" @click="handleOpenTransfer">
103
+ {{ i18nText('1.10.201') }}
104
+ </div>
105
+ <div class="toSession primary" @click="handleAccess">
106
+ {{ accessText || i18nText('1.9.352') }}
107
+ </div>
108
+ <div v-if="enableRefuse" class="toSession danger" @click="openRefuseReason">
109
+ {{ i18nText('1.9.353') }}
110
+ </div>
111
+ </div>
112
+ </div>
113
+ <!-- 操作栏弹窗 -->
114
+ <a-modal
115
+ :wrapClassName="isFullScreen ? 'chat-footer-modal-big' : 'chat-footer-modal-small'"
116
+ :title="modalData.name"
117
+ :visible="modalShow"
118
+ :maskClosable="false"
119
+ :mask="true"
120
+ :width="isFullScreen ? '100%' : '80%'"
121
+ destroyOnClose
122
+ :okText="i18nText('1.2.1.11.6')"
123
+ :footer="null"
124
+ @cancel="handleClose"
125
+ @ok="handleOk"
126
+ >
127
+ <iframe id="toolbarIframe" v-if="modalData.targetType === 'LINK_ADDRESS'" width="100%" height="100%" frameborder="0" :src="modalData.address" allow="camera;midi"></iframe>
128
+ </a-modal>
129
+ <!-- 接入拦截 -->
130
+ <a-modal
131
+ :title="accessInterceptTitle"
132
+ :visible="accessIntercept"
133
+ :maskClosable="false"
134
+ :mask="true"
135
+ :footer="null"
136
+ destroyOnClose
137
+ :width="600"
138
+ :bodyStyle="{ height: '280px' }"
139
+ @cancel="
140
+ () => {
141
+ this.accessIntercept = false;
142
+ }
143
+ "
144
+ >
145
+ <iframe width="100%" height="100%" frameborder="0" :src="accessInterceptUrl"></iframe>
146
+ </a-modal>
147
+ <!-- 语音消息 -->
148
+ <voice v-if="showVoice" :showVoice="showVoice" @closeVoice="handleVoice"></voice>
149
+ <ModalUserTransfer v-if="enableTransfer" :title="i18nText('1.10.201')" :visible.sync="transferVisible" :assemblyId="assemblyId" @ok="handleTransfer" />
150
+ <ModalRefuseReason v-if="refuseVisible" :visible.sync="refuseVisible" :reasonList="refuseReasonList" @ok="handleRefuse" />
151
+ </a-layout-footer>
152
+ </template>
153
+
154
+ <script>
155
+ import { Layout, Tooltip, Upload, Modal, Popover, Button } from 'ant-design-vue';
156
+ import SvgIcon from '@/component/svg/index.vue';
157
+ import ImageCompress from '../utils/compressImage';
158
+ import { parseParams } from '../utils/panelsetting';
159
+
160
+ import { mapState, mapGetters, mapMutations, mapActions } from '../store/helper';
161
+ import fetch, { qs } from '@/utils/chatFetch';
162
+ import QuickReply from './quickReply';
163
+ import voice from './voice';
164
+ import ModalUserTransfer from '../components/modal-user-transfer';
165
+ import ModalRefuseReason from '../components/modal-refuse-reason';
166
+ import vexutils from '@/utils/vexutils';
167
+ export default {
168
+ inject: ['store', 'i18nText', 'dispatchEvent', 'registerEvent', 'unregisterEvent'],
169
+ components: {
170
+ AModal: Modal,
171
+ [Layout.Footer.name]: Layout.Footer,
172
+ [Tooltip.name]: Tooltip,
173
+ [Upload.name]: Upload,
174
+ [Button.name]: Button,
175
+ [Popover.name]: Popover,
176
+ SvgIcon,
177
+ ModalRefuseReason,
178
+ ModalUserTransfer,
179
+ QuickReply,
180
+ voice
181
+ },
182
+ props: {
183
+ // 隐藏排队状态底部信息
184
+ hideQueueBtns: {
185
+ type: Boolean,
186
+ default: false
187
+ },
188
+ validTalkingEnv: {
189
+ type: Function
190
+ },
191
+ curChatType: {
192
+ type: String
193
+ },
194
+ clientFinish: {
195
+ type: Boolean,
196
+ default: true
197
+ },
198
+ addvisible: {
199
+ type: Boolean,
200
+ default: false
201
+ },
202
+ isShowVoice: {
203
+ type: Boolean,
204
+ default: false
205
+ }
206
+ },
207
+ data() {
208
+ return {
209
+ // message: "",
210
+ visible: false,
211
+ modalShow: false,
212
+ modalData: {},
213
+ ping: false,
214
+ templateId: null, // 卡片模板
215
+ isFullScreen: false,
216
+ emojiList: require('../utils/emoji.json'),
217
+ quickData: [],
218
+ quickShow: false,
219
+ screenshot: false,
220
+ screenshotFile: null,
221
+ timer: null,
222
+ accessIntercept: false,
223
+ curToolbarItem: {},
224
+ showVoice: false,
225
+ transferVisible: false,
226
+ refuseVisible: false,
227
+
228
+ apiResult: {}, // 前置条件的返回值
229
+ };
230
+ },
231
+ computed: {
232
+ ...mapGetters([
233
+ 'userInfo',
234
+ 'assemblyId',
235
+ 'sessionId',
236
+ 'serviceId',
237
+ 'onChating',
238
+ 'roomId',
239
+ 'isServer',
240
+ 'assemblySetting',
241
+ 'chatType',
242
+ 'sessionEnd',
243
+ 'closedSession',
244
+ 'clientParams',
245
+ 'footerMessage',
246
+ 'queueItem',
247
+ 'sessionType',
248
+ 'groupMembers',
249
+ 'curScrollItem'
250
+ ]),
251
+ ...mapState({
252
+ classify: state => state.lastClassify || state.classify
253
+ }),
254
+ message() {
255
+ return this.footerMessage;
256
+ },
257
+ sendDisabled() {
258
+ return this.message.length === 0;
259
+ },
260
+ orgId() {
261
+ return this.userInfo?.sysParams?.orgId;
262
+ },
263
+ userId() {
264
+ return this.userInfo?.sysParams?.userId;
265
+ },
266
+ toolbar() {
267
+ if (!this.isServer) return [];
268
+ let toolbar = this.assemblySetting.toolbar || [];
269
+ toolbar = toolbar.filter(item => {
270
+ const { source, isChecked, isCloseChecked, isClosedAble } = item;
271
+ if (source === 'PC' && (
272
+ (!this.closedSession && isChecked === 'Y') ||
273
+ (this.closedSession && isCloseChecked === 'Y' && isChecked === 'Y' && isClosedAble === 'Y')
274
+ )
275
+ ){
276
+ const showCondition = item.showCondition || [];
277
+ const i = showCondition.findIndex(({ p_name, p_value }) => {
278
+ const params = this.clientParams;
279
+ const value = params[p_name];
280
+ if (p_value.startsWith('form.')) {
281
+ return value !== p_name[p_value.slice(5)];
282
+ }
283
+ return value !== p_value;
284
+ });
285
+ return i < 0;
286
+ }
287
+ return false;
288
+ });
289
+ return toolbar;
290
+ },
291
+ isListClassify() {
292
+ return this.assemblySetting.isListClassify === 'Y';
293
+ },
294
+ showOverlay() {
295
+ return this.curChatType != 'robot' || this.clientFinish;
296
+ },
297
+ accessText() {
298
+ let obj = this.assemblySetting.serverSetting.serverInfo.filter(item => item.key === 'ongoingButton')[0];
299
+ return obj?.value;
300
+ },
301
+ accessDesc() {
302
+ let obj = this.assemblySetting.serverSetting.serverInfo.filter(item => item.key === 'ongoingDescription')[0];
303
+ return obj?.value;
304
+ },
305
+ accessInterceptObj() {
306
+ let obj = this.assemblySetting.accessInterceptSetting?.filter(item => item.source === 'PC');
307
+ if (obj && obj[0]) {
308
+ return obj[0];
309
+ }
310
+ return undefined;
311
+ },
312
+ // 拦截url
313
+ accessInterceptUrl() {
314
+ const { accessInterceptObj: obj, apiResult } = this;
315
+ if (obj) {
316
+ return this.getLinkAddress(obj,{result:apiResult});
317
+ }
318
+ return '';
319
+ },
320
+ // 拦截title
321
+ accessInterceptTitle() {
322
+ const { accessInterceptObj: obj } = this;
323
+ if (obj) {
324
+ let res = obj.titleI18n ? this.i18nText(obj.titleI18n) : obj.title;
325
+ return res || '接入拦截';
326
+ }
327
+ return '接入拦截';
328
+ },
329
+ // 是否开启 排队转接
330
+ enableTransfer() {
331
+ if (this.assemblySetting.type === 'single') {
332
+ let setting = null;
333
+ if (this.isListClassify) {
334
+ const classify = this.classify;
335
+ const { listClassify = [] } = this.assemblySetting;
336
+ const target = listClassify.find(item => item.classify === classify);
337
+ setting = target ? target.serverFunctionSetting : null;
338
+ } else {
339
+ setting = this.assemblySetting?.serverSetting?.functionSetting;
340
+ }
341
+ if (setting) {
342
+ const v = setting.find(item => item.function === 'TRANSFER_BEFORE');
343
+ return v ? v.isChecked == 'Y' : false;
344
+ }
345
+ }
346
+ return false;
347
+ },
348
+ enableRefuse() {
349
+ let setting = null;
350
+ if (this.isListClassify) {
351
+ const classify = this.classify;
352
+ const { listClassify = [] } = this.assemblySetting;
353
+ const target = listClassify.find(item => item.classify === classify);
354
+ setting = target ? target.serverFunctionSetting : null;
355
+ } else {
356
+ setting = this.assemblySetting?.serverSetting?.functionSetting;
357
+ }
358
+ if (setting) {
359
+ const v = setting.find(item => item.function === 'ACCESS_REFUSE');
360
+ return v ? v.isChecked == 'Y' : false;
361
+ }
362
+ return false;
363
+ },
364
+ refuseReasonList() {
365
+ const list = this.assemblySetting?.refuseReasonSetting?.list || [];
366
+ return list.filter(v => !!v.text);
367
+ }
368
+ },
369
+
370
+ mounted() {
371
+ this.bindEvents();
372
+ },
373
+ methods: {
374
+ ...mapMutations([
375
+ 'setLastClassify',
376
+ 'setSessionTimer',
377
+ 'setScrollQueueId',
378
+ 'setCurScrollItem',
379
+ 'setIsAppendMsg',
380
+ 'setMessage',
381
+ 'setShowVideo',
382
+ 'setCaller',
383
+ 'setMedicalOrPrescription',
384
+ 'setModalVisible',
385
+ 'setFooterMessage',
386
+ 'setQueueItem',
387
+ 'setSessionId',
388
+ 'setCurrentTab',
389
+ 'setClientId',
390
+ 'setOnChating',
391
+ 'setClientParams',
392
+ 'setSessionHistoryList',
393
+ 'setSessionEnd',
394
+ 'setQuestionId',
395
+ 'setDictionaryKey',
396
+ 'setMultiVideoShow',
397
+ 'setVideoStatus',
398
+ 'setVideoMode',
399
+ 'setAddMemberType',
400
+ 'setShowAudio',
401
+ 'setVideoMembers',
402
+ 'setLastCurrentTab',
403
+ 'setIsRecorderVoice',
404
+ 'setVideoData',
405
+ 'setSessionType',
406
+ 'setCallerStatus'
407
+ ]),
408
+ ...mapMutations({
409
+ clearMsgList: 'setMsgList'
410
+ }),
411
+ ...mapActions(['sendMessage', 'sendRobotMessage', 'handleBotChat', 'setMsgList', 'setChatTimer']),
412
+ deleteStyle(value) {
413
+ // 去掉样式
414
+ value = value.replace(/style\s*?=\s*?(['"])[\s\S]*?\1/g, '');
415
+ value = value.replace(/data\s*?=\s*?(['"])[\s\S]*?\1/g, '');
416
+ value = value.replace(/class\s*?=\s*?(['"])[\s\S]*?\1/g, match => {
417
+ if (match.includes('twa')) {
418
+ return match;
419
+ } else {
420
+ return '';
421
+ }
422
+ });
423
+ value = value.replace(/face\s*?=\s*?(['"])[\s\S]*?\1/g, '');
424
+ value = value.replace(/<a/g, '<span');
425
+ value = value.replace(/a>/g, 'span>');
426
+ value = value.replace(/<i/g, '<span');
427
+ value = value.replace(/i>/g, 'span>');
428
+ value = value.replace(/<u/g, '<span');
429
+ value = value.replace(/u>/g, 'span>');
430
+ value = value.replace(/<strong/g, '<span');
431
+ value = value.replace(/strong>/g, 'span>');
432
+ value = value.replace(/<h\d/g, '<span');
433
+ value = value.replace(/h\d>/g, 'span>');
434
+ value = value.replace(/<b>/g, '<span>');
435
+ value = value.replace(/b>/g, 'span>');
436
+ value = value.replace(/<font/g, '<span');
437
+ value = value.replace(/font>/g, 'span>');
438
+ return value;
439
+ },
440
+ // 更新 message值 (未用v-html,修改后需要手动设置聊天框html内容)
441
+ updateMessage(value) {
442
+ this._editValue = value;
443
+ this.setFooterMessage(`${value}`);
444
+ const editRef = this.$refs.contentEdit;
445
+ if (!editRef) return;
446
+ editRef.innerHTML = value;
447
+ },
448
+ // 根据 聊天框 html节点内容 设置 message值 (未用v-html,html内容更改需主动设置message值)
449
+ setMessageByHTML() {
450
+ const editRef = this.$refs.contentEdit;
451
+ if (!editRef) return;
452
+ const message = editRef.innerHTML;
453
+ this._editValue = message;
454
+ this.setFooterMessage(`${message}`);
455
+ },
456
+
457
+ showEmoji() {
458
+ // this.visible = true;
459
+ },
460
+
461
+ isShow(type) {
462
+ if (type === 'SMILINEFACE' && this.curChatType === 'robot') return true;
463
+ if (this.isListClassify) {
464
+ let label;
465
+ if (this.isServer) {
466
+ label = 'serverFunctionSetting';
467
+ } else {
468
+ label = 'clientFunctionSetting';
469
+ }
470
+ let classifyItem = this.assemblySetting.listClassify.filter(item => item.classify === this.classify);
471
+ if (!classifyItem || !classifyItem.length) return;
472
+ let funs = classifyItem[0][label] || [];
473
+ return funs.find(item => item.function === type)?.isChecked === 'Y';
474
+ }
475
+ let label;
476
+ if (this.isServer) {
477
+ label = 'serverSetting';
478
+ } else {
479
+ label = 'clientSetting';
480
+ }
481
+ let funs = this.assemblySetting[label]?.functionSetting || [];
482
+ return funs.find(item => item.function === type)?.isChecked === 'Y';
483
+ },
484
+
485
+ handleEmojiClick(item) {
486
+ let emojiHtml = `<p contenteditable="false" class="twa emoji ${item}"></p>&nbsp`;
487
+ this.insertToEdit(emojiHtml);
488
+ },
489
+ /* edit-content光标定位 */
490
+ resetSelection() {
491
+ let edit = this.$refs.contentEdit;
492
+ let sel, range;
493
+ if (window.getSelection && document.createRange) {
494
+ range = document.createRange();
495
+ range.selectNodeContents(edit);
496
+ range.collapse(true);
497
+ range.setEnd(edit, edit.childNodes.length);
498
+ range.setStart(edit, edit.childNodes.length);
499
+ sel = window.getSelection();
500
+ sel.removeAllRanges();
501
+ sel.addRange(range);
502
+ }
503
+ },
504
+
505
+ insertToEdit(value, isEditing) {
506
+ const { _editSel, _editRange } = this;
507
+ let sel, range;
508
+ if (window.getSelection) {
509
+ if (!isEditing) {
510
+ if (_editRange) {
511
+ sel = _editSel;
512
+ range = _editRange;
513
+ } else {
514
+ // 初始状态没有focus聊天框,需要手动focus 获取光标位置
515
+ const editRef = this.$refs.contentEdit;
516
+ if (!editRef) return;
517
+ editRef.focus();
518
+ this.saveRange();
519
+ sel = this._editSel;
520
+ range = this._editRange;
521
+ }
522
+ } else {
523
+ sel = window.getSelection();
524
+ range = sel.getRangeAt(0);
525
+ }
526
+ range.deleteContents();
527
+ let frag;
528
+ if (range.createContextualFragment) {
529
+ frag = range.createContextualFragment(value);
530
+ } else {
531
+ const el = document.createElement('div');
532
+ el.innerHTML = value;
533
+ frag = document.createDocumentFragment();
534
+ let node, lastNode;
535
+ while ((node = el.firstChild)) {
536
+ lastNode = frag.appendChild(node); // 之所以能终止循环,是因为el的child一旦添加到dom后,作为fragment里的的child就没了。那么el.firstChild就会返回null
537
+ }
538
+ }
539
+ const lastNode = frag.lastChild;
540
+ range.insertNode(frag);
541
+ // 把光标挪到插入的元素后面
542
+ range.setStartAfter(lastNode);
543
+ sel.removeAllRanges();
544
+ sel.addRange(range);
545
+ } else {
546
+ range = isEditing || !_editRange ? document.selection.createRange() : _editRange;
547
+ range.pasteHTML(value);
548
+ range.select();
549
+ }
550
+ this.setMessageByHTML();
551
+ },
552
+ saveRange() {
553
+ if (window.getSelection) {
554
+ this._editSel = window.getSelection();
555
+ this._editRange = this._editSel.getRangeAt(0);
556
+ } else {
557
+ this._editRange = document.selection.createRange();
558
+ }
559
+ },
560
+ handleInput(e) {
561
+ if (e.inputType === 'deleteContentBackward') {
562
+ this.screenshot = false;
563
+ this.screenshotFile = null;
564
+ }
565
+ setTimeout(() => {
566
+ if (this.ping || this.pasteTimer) return;
567
+ this.setMessageByHTML();
568
+ }, 0);
569
+ },
570
+ changeData(e) {
571
+ let message = e.srcElement.innerHTML;
572
+ if (e.inputType === 'deleteContentBackward') {
573
+ let div = document.createElement('div');
574
+ div.innerHTML = e.srcElement.innerHTML;
575
+ if (div.lastChild?.className?.startsWith('twa emoji')) {
576
+ div.removeChild(div.lastChild);
577
+ message = div.innerHTML;
578
+ }
579
+ this.screenshot = false;
580
+ this.screenshotFile = null;
581
+ }
582
+
583
+ setTimeout(() => {
584
+ if (this.ping) return;
585
+ this.setFooterMessage(`${message}`);
586
+ this.$nextTick().then(() => {
587
+ this.resetSelection();
588
+ this.ping = false;
589
+ });
590
+ }, 0);
591
+ },
592
+
593
+ beforeUpload(data) {
594
+ let { type } = data;
595
+ if (!type.startsWith('image/')) {
596
+ this.$message.warning(`${this.i18nText('1.9.357')}!`);
597
+ return;
598
+ }
599
+ let that = this;
600
+ this.$confirm({
601
+ title: this.i18nText('1.9.331'),
602
+ content: `${this.i18nText('1.9.358')}?`,
603
+ okText: '确定',
604
+ cancelText: '取消',
605
+ onOk() {
606
+ return that.uploadImg(data);
607
+ },
608
+ onCancel() {
609
+ return false;
610
+ }
611
+ });
612
+ },
613
+
614
+ /* 压缩上传 */
615
+ uploadImg(file) {
616
+ const hide = this.$message.loading(`${this.i18nText('1.9.199')}...`, 0);
617
+ new ImageCompress(file, 'canvas', fetch, {
618
+ url: '/hospitalUpload/picture'
619
+ })
620
+ .execute()
621
+ .then(({ data }) => {
622
+ if (data.result === 'SUCCESS') {
623
+ this.doMsgSend(1, data.map.url);
624
+ hide();
625
+ } else {
626
+ hide();
627
+ this.$message.info(data.msg || `${this.i18nText('1.9.174')}!`);
628
+ }
629
+ })
630
+ .catch(() => {
631
+ hide();
632
+ this.$message.info(`${this.i18nText('1.9.174')}!`);
633
+ })
634
+ .finally(() => {
635
+ this.screenshotFile = null;
636
+ this.screenshot = false;
637
+ this.updateMessage('');
638
+ });
639
+ },
640
+
641
+ /* 发送消息 */
642
+ doMsgSend(type, content) {
643
+ let body = {
644
+ content: type === 0 ? `<div>${content}</div>` : content,
645
+ type: type
646
+ };
647
+ if (this.curChatType === 'robot') {
648
+ // this.setQuestionId("");
649
+ // this.setDictionaryKey("");
650
+ this.sendRobotMessage(content);
651
+ } else {
652
+ this.sendMessage(body);
653
+ }
654
+ },
655
+ handleMsgSend() {
656
+ // 截图上传
657
+ if (this.screenshot) {
658
+ this.uploadImg(this.screenshotFile);
659
+ return;
660
+ }
661
+ if (this.chatType === 'bot') {
662
+ this.handleBotChat(this.message).then(isHuman => {
663
+ if (isHuman) {
664
+ this.$parent.$parent.$parent.initWebSocket();
665
+ }
666
+ this.updateMessage('');
667
+ });
668
+ return;
669
+ }
670
+
671
+ const editRef = this.$refs.contentEdit;
672
+ if (editRef) {
673
+ const message = editRef.innerText;
674
+ if (message.length > 2000) {
675
+ this.$message.warning('请控制在2000字以内!');
676
+ return;
677
+ }
678
+ }
679
+ // let reg = /(.*?)(<div><br><\/div>)*?$/;
680
+ let reg = /(.*?)(<div>(<span>)?<br>(<\/span>)?<\/div>)*?$/;
681
+ let message = this.message.match(reg);
682
+ if (message[2]) {
683
+ this.updateMessage(message[1]);
684
+ }
685
+ this.updateMessage(this.message.replace(/style/g, 'data'));
686
+ if (this.message.trim()) {
687
+ this.doMsgSend(0, this.message);
688
+ } else {
689
+ return this.$message.warning(this.i18nText('1.9.359'));
690
+ }
691
+ this.updateMessage('');
692
+ },
693
+ handleKeyDown(e) {
694
+ // 判断enter 发送还是换行
695
+ if (e.keyCode === 13) {
696
+ e.preventDefault();
697
+ if (e.altKey || e.ctrlKey || e.metaKey || e.shiftKey) {
698
+ this.handleWrap();
699
+ } else {
700
+ this.handleMsgSend();
701
+ }
702
+ }
703
+ },
704
+ handleWrap() {
705
+ let ctn = `<br>`;
706
+ // 非ie环境 元素后面没有node的时候换行需要两个<br>
707
+ if (!/msie/.test(navigator.userAgent) && window.getSelection) {
708
+ const focusNode = window.getSelection().focusNode;
709
+ let nextSibling = focusNode.nextSibling;
710
+ if (focusNode === this.$refs.contentEdit) {
711
+ nextSibling = focusNode.childNodes[0];
712
+ }
713
+ do {
714
+ if (!nextSibling || nextSibling.nodeValue || nextSibling.tagName == 'BR') break;
715
+ } while ((nextSibling = nextSibling.nextSibling));
716
+ if (!nextSibling) {
717
+ ctn += ctn;
718
+ }
719
+ }
720
+ this.insertToEdit(ctn);
721
+ this.$nextTick().then(() => {
722
+ this.ping = false;
723
+ });
724
+ },
725
+ handleVideoClick(type) {
726
+ this.setCallerStatus('')
727
+ // this.curScrollItem.type 2多人 1单人
728
+ if (this.sessionType === 2) {
729
+ this.$emit('update:addvisible', true);
730
+ this.setAddMemberType(2);
731
+ this.setVideoMode(type === 'voice' ? 1 : 2);
732
+ } else {
733
+ const openTalking = sip => {
734
+ this.setShowAudio(true);
735
+ this.setVideoMode(type === 'voice' ? 1 : 2);
736
+ this.$nextTick(() => {
737
+ this.setVideoMembers([]);
738
+ });
739
+ this.setVideoData({
740
+ callerSipKey: sip?.username,
741
+ callerSipName: sip?.name
742
+ });
743
+ };
744
+ if (this.validTalkingEnv) {
745
+ this.validTalkingEnv().then(res => {
746
+ if (res.status) {
747
+ openTalking(res.sip);
748
+ }
749
+ });
750
+ } else {
751
+ openTalking();
752
+ }
753
+ }
754
+ // this.groupMembers.length > 2
755
+ return;
756
+ },
757
+ handleVoice(val) {
758
+ val && this.setIsRecorderVoice(true);
759
+ this.showVoice = val;
760
+ },
761
+ showPrescription() {
762
+ this.setModalVisible(true);
763
+ this.setMedicalOrPrescription('prescription');
764
+ },
765
+ showMedicalRecord() {
766
+ this.setModalVisible(true);
767
+ this.setMedicalOrPrescription('medical');
768
+ },
769
+ beofreToolBarClick(item) {
770
+ // 操作栏前置条件
771
+ return new Promise((resolve, reject) => {
772
+ const { precondition } = item;
773
+ if (precondition) {
774
+ this.requestPreCondition(precondition)
775
+ .then(res => {
776
+ const result = res.data.result === 'SUCCESS';
777
+ !result && this.$message.error(res.data.resultMsg || this.i18nText('1.9.22'));
778
+ resolve(result);
779
+ })
780
+ .catch(error => {
781
+ this.$message.error(error.resultMsg || this.i18nText('1.9.22'));
782
+ resolve(false);
783
+ });
784
+ } else {
785
+ resolve(true);
786
+ }
787
+ });
788
+ },
789
+ async handleToolBarClick(item) {
790
+ const validate = await this.beofreToolBarClick(item);
791
+ if (!validate) return;
792
+ let { openMode, name, templateId, isFullScreen } = item;
793
+ this.dispatchEvent('click_toolbar', { ...item });
794
+ if (!item.address) return;
795
+ this.curToolbarItem = item;
796
+ this.templateId = templateId;
797
+ this.isFullScreen = isFullScreen === 'Y';
798
+ let address = this.getLinkAddress(item);
799
+ if (openMode === 'WINDOW') {
800
+ window.open(address, name);
801
+ return;
802
+ }
803
+ if (openMode === 'EJECT') {
804
+ this.modalShow = true;
805
+ this.modalData = Object.assign({}, item, { address });
806
+ this.$nextTick().then(() => {
807
+ window.addEventListener('message', this.iframeEvent);
808
+ });
809
+ return;
810
+ }
811
+ },
812
+ iframeEvent(event) {
813
+ const method = event?.data?.method;
814
+ this[method]?.(event);
815
+ },
816
+ // 发送
817
+ handleParentMessageSend({ data, source, origin }) {
818
+ let params = {
819
+ assemblyId: this.assemblyId,
820
+ sessionId: this.sessionId,
821
+ orgId: this.orgId
822
+ };
823
+ params = Object.assign(params, data.data);
824
+ let { actionType } = this.curToolbarItem;
825
+ actionType && Object.assign(params, { actionType });
826
+ return fetch.post('/chat/access/sendToolBarData', qs.stringify(params)).then(({ data }) => {
827
+ if (data.result === 'SUCCESS') {
828
+ source.postMessage({ status: 0, resultMsg: data.resultMsg }, origin);
829
+ } else {
830
+ source.postMessage({ status: 1, resultMsg: data.resultMsg }, origin);
831
+ }
832
+ });
833
+ },
834
+ // 关闭
835
+ handleClose() {
836
+ this.modalShow = false;
837
+ window.removeEventListener('message', this.iframeEvent);
838
+ },
839
+ getLinkAddress(item,paramsData= {}) {
840
+ let t = {talbe:{},form:{},sys:{}};
841
+ Object.assign(t,paramsData ||{})
842
+ let { address = '', params = [] } = item;
843
+ let urlParams = [];
844
+ params.forEach(({ p_name, p_value }) => {
845
+ let value;
846
+ if (p_value.startsWith('form.')) {
847
+ value = this.clientParams[p_value.slice(5)];
848
+ } else if( /\${result.(.*?)\}/g.test(p_value)) {
849
+ value = vexutils.handleSysParams(p_value,t)
850
+ } else {
851
+ value = p_value;
852
+ }
853
+ urlParams.push(`${p_name}=${value}`);
854
+ });
855
+ if(/\${result.(.*?)\}/g.test(address)){
856
+ address = vexutils.handleSysParams(address,t)
857
+ }
858
+ let { actionType } = this.curToolbarItem;
859
+ actionType && urlParams.push(`actionType=${actionType}`);
860
+ if (address.includes('?')) {
861
+ address += `&${urlParams.join('&')}`;
862
+ } else {
863
+ address += `?${urlParams.join('&')}`;
864
+ }
865
+ return address;
866
+ },
867
+ handleOk() {
868
+ window.handleParentMessageSend();
869
+ },
870
+ // 剪切板粘贴
871
+ handlePaste(e) {
872
+ const cbd = e.clipboardData;
873
+ const ua = window.navigator.userAgent;
874
+ // 如果是 Safari 直接 return
875
+ if (!(e.clipboardData && e.clipboardData.items)) {
876
+ this.handlePasteText(e);
877
+ return;
878
+ }
879
+ if (
880
+ cbd.items &&
881
+ cbd.items.length === 2 &&
882
+ cbd.items[0].kind === 'string' &&
883
+ cbd.items[1].kind === 'file' &&
884
+ cbd.types &&
885
+ cbd.types.length === 2 &&
886
+ cbd.types[0] === 'text/plain' &&
887
+ cbd.types[1] === 'Files' &&
888
+ ua.match(/Macintosh/i) &&
889
+ Number(ua.match(/Chrome\/(\d{2})/i)[1]) < 49
890
+ ) {
891
+ this.handlePasteText(e);
892
+ return;
893
+ }
894
+ const hasFile = this.handleImgPaste(cbd.items);
895
+ if (hasFile) {
896
+ e.stopPropagation();
897
+ e.preventDefault();
898
+ return;
899
+ }
900
+ this.handlePasteText(e);
901
+ },
902
+ handleImgPaste(files) {
903
+ let hasFile = false;
904
+ for (let i = 0; i < files.length; i++) {
905
+ let item = files[i];
906
+ if (item.kind == 'file') {
907
+ hasFile = true;
908
+ // blob 就是从剪切板获得的文件,可以进行上传或其他操作
909
+ const blob = item.getAsFile();
910
+ if (blob.size === 0) {
911
+ return;
912
+ }
913
+ const reader = new FileReader();
914
+ const imgs = new Image();
915
+ imgs.file = blob;
916
+ reader.onload = e => {
917
+ imgs.src = e.target.result;
918
+ imgs.width = '150';
919
+ this.updateMessage(this.htmlToString(imgs));
920
+ this.$nextTick().then(() => {
921
+ this.resetSelection();
922
+ this.screenshot = true;
923
+ this.screenshotFile = blob;
924
+ });
925
+ };
926
+ reader.readAsDataURL(blob);
927
+ }
928
+ }
929
+ return hasFile;
930
+ },
931
+ // 处理文本粘贴
932
+ handlePasteText() {
933
+ if (this.pasteTimer) clearTimeout(this.pasteTimer);
934
+ const ele = this.$refs.contentEdit;
935
+ if (!ele) return;
936
+ let before = ele.innerHTML;
937
+ this.pasteTimer = setTimeout(() => {
938
+ function getPasteHtml(before, after) {
939
+ // 计算粘贴后html不同的位置起点和终点(插入"abc",则得出sPos为0,ePos为2)
940
+ var sPos = -1,
941
+ ePos = -1;
942
+ for (var i = 0, len = after.length; i < len; ++i) {
943
+ if (sPos == -1 && before.substr(i, 1) != after.substr(i, 1)) sPos = i;
944
+ if (ePos == -1 && before.substr(before.length - i - 1, 1) != after.substr(after.length - i - 1, 1)) ePos = i;
945
+
946
+ if ((sPos != -1 && ePos != -1) || len - 1 - ePos <= sPos) break;
947
+ }
948
+ if (sPos == -1 || ePos == -1) return;
949
+ ePos = len - 1 - ePos;
950
+
951
+ if (ePos <= sPos) {
952
+ // 遇到像ab,粘贴ab后变成abab这种了,这是需要从sPos开始往后找,找到和before一样的为止
953
+ i = sPos;
954
+ var beforeSPosNext = before.substr(i + 1, 10); // 取后10个来对比,如果没有10个,则取最多
955
+ while (++i < len) {
956
+ if (beforeSPosNext == after.substr(i, beforeSPosNext.length)) {
957
+ ePos = i;
958
+ break;
959
+ }
960
+ }
961
+ i == len && (ePos = len - 1);
962
+ }
963
+ // 如果有<和>,则需要计算在内
964
+ if (after.substr(sPos - 1, 1) == '<') --sPos;
965
+ if (after.substr(ePos + 1, 1) == '>') ++ePos;
966
+
967
+ // 分段获得文本,前半段 插入段 后半段
968
+ var pastedText = after.substring(sPos, ePos + 1),
969
+ formerText = after.substr(0, sPos),
970
+ latterText = after.substr(sPos + pastedText.length);
971
+
972
+ // 判断formerText和pastedText是否闭合了
973
+ var lastLT = formerText.lastIndexOf('<'),
974
+ lastGT = formerText.lastIndexOf('>');
975
+ if (lastGT < lastLT) {
976
+ // 标签断了xxxxx<
977
+ pastedText = formerText.slice(lastLT) + pastedText;
978
+ formerText = formerText.slice(0, lastLT);
979
+ }
980
+
981
+ lastLT = pastedText.lastIndexOf('<');
982
+ lastGT = pastedText.lastIndexOf('>');
983
+ if (lastGT < lastLT) {
984
+ // 标签断了<xxxxxx
985
+ var _insertGT = latterText.indexOf('>') + 1;
986
+ pastedText += latterText.slice(0, _insertGT);
987
+ latterText = latterText.slice(_insertGT);
988
+ }
989
+ return [formerText, pastedText, latterText];
990
+ }
991
+ var after = ele.innerHTML;
992
+ const data = getPasteHtml(before, after);
993
+ if (data) {
994
+ let [formerText, insertText, latterText] = data;
995
+ insertText = this.deleteStyle(insertText);
996
+ ele.innerHTML = formerText + insertText + "<span class='pasteCaretPosHelper'></span>" + latterText; // 插入pasteCaretPosHelper以帮助定位光标位置
997
+ var pasteCaretPosHelper = ele.querySelector('.pasteCaretPosHelper'),
998
+ range,
999
+ selection;
1000
+ if (pasteCaretPosHelper) {
1001
+ if (document.createRange) {
1002
+ // Chrome, IE 9+
1003
+ range = document.createRange();
1004
+ range.setStartAfter(pasteCaretPosHelper);
1005
+ range.collapse(false);
1006
+ selection = window.getSelection();
1007
+ selection.removeAllRanges();
1008
+ selection.addRange(range);
1009
+ } else if (document.selection) {
1010
+ // IE 8 and lower
1011
+ range = document.body.createTextRange();
1012
+ range.moveToElementText(pasteCaretPosHelper);
1013
+ range.collapse(false);
1014
+ range.select();
1015
+ }
1016
+ pasteCaretPosHelper.parentNode.removeChild(pasteCaretPosHelper);
1017
+ }
1018
+ }
1019
+ this.setMessageByHTML();
1020
+ this.saveRange();
1021
+ this.pasteTimer = null;
1022
+ }, 50);
1023
+ },
1024
+ // 拖拽上传图片
1025
+ handleFiles(files) {
1026
+ this.handleImgPaste(files);
1027
+ },
1028
+ htmlToString(dom) {
1029
+ let div = document.createElement('div');
1030
+ div.appendChild(dom);
1031
+ return div.innerHTML;
1032
+ },
1033
+ /**
1034
+ * 接入拦截
1035
+ */
1036
+ handleAccessIntercept(result) {
1037
+ let { data } = result;
1038
+ let { status, data: param } = data;
1039
+ if (status === 1) {
1040
+ this.accessChat(param).finally(() => {
1041
+ this.accessIntercept = false;
1042
+ });
1043
+ } else {
1044
+ this.accessIntercept = false;
1045
+ }
1046
+ },
1047
+ /* 客服点击接入按钮进入会话 */
1048
+ async handleAccess() {
1049
+ this.apiResult = {};
1050
+ let accessInterceptValidate = this.handleAccessInterceptValue();
1051
+ let data = null; // 接口参数
1052
+ if (accessInterceptValidate && this.accessInterceptUrl) {
1053
+ const r = await this.beforeAccessIntercept();
1054
+ if (!r) return;
1055
+ if (r.value == 'Y') {
1056
+ this.accessIntercept = true;
1057
+ this.$nextTick().then(() => {
1058
+ window.addEventListener('message', this.handleAccessIntercept);
1059
+ });
1060
+ return;
1061
+ } else {
1062
+ data = r.data; // 获取接入拦截返回参数
1063
+ }
1064
+ }
1065
+ clearTimeout(this.timer);
1066
+ this.timer = setTimeout(() => {
1067
+ this.accessChat(data);
1068
+ }, 300);
1069
+ },
1070
+ // 条件转换请求
1071
+ requestPreCondition(precondition) {
1072
+ const { targetType, targetName, targetId, params } = precondition;
1073
+ const data = parseParams(params, this.clientParams);
1074
+ let url = '';
1075
+ if (targetType === 'API') {
1076
+ data.url = targetName;
1077
+ url = `/api/executeApi`;
1078
+ } else {
1079
+ data.sqlEnterId = targetId;
1080
+ url = `/form/saveSqlEnter`;
1081
+ }
1082
+ return fetch.post(url, qs.stringify(data));
1083
+ },
1084
+ async beforeAccessIntercept() {
1085
+ function parseResult(r) {
1086
+ let value = 'Y';
1087
+ const data = r.split('&').reduce((obj, str) => {
1088
+ const [k, v] = str.split('=');
1089
+ if (k) {
1090
+ if (k == 'value') {
1091
+ value = v || value;
1092
+ } else {
1093
+ obj = obj || {};
1094
+ obj[k] = v || '';
1095
+ }
1096
+ }
1097
+ return obj;
1098
+ }, null);
1099
+ return {
1100
+ value,
1101
+ data
1102
+ };
1103
+ }
1104
+ this.apiResult = {};
1105
+ const accessInterceptSetting = this.assemblySetting.accessInterceptSetting || [];
1106
+ const setting = accessInterceptSetting.find(item => item.source == 'PC' && item.isChecked == 'Y' && item.precondition);
1107
+ if (setting) {
1108
+ const precondition = setting.precondition;
1109
+ try {
1110
+ let res = await this.requestPreCondition(precondition);
1111
+ this.apiResult = res?.data?.map || {};
1112
+ if (res.data.result === 'SUCCESS') {
1113
+ let value = res.data.map?.result;
1114
+ value = value ? JSON.parse(value) : {};
1115
+ return value.value ? parseResult(value.value) : false;
1116
+ } else {
1117
+ this.$message.error(res.data.resultMsg || this.i18nText('1.9.22'));
1118
+ return false;
1119
+ }
1120
+ } catch (error) {
1121
+ this.$message.error(error.resultMsg || this.i18nText('1.9.22'));
1122
+ return false;
1123
+ }
1124
+ }
1125
+ return {
1126
+ value: 'Y',
1127
+ data: null
1128
+ };
1129
+ },
1130
+ handleAccessInterceptValue() {
1131
+ let { listClassify = [], isListClassify, serverSetting = {} } = this.assemblySetting;
1132
+ let list = [];
1133
+ if (isListClassify != 'Y') {
1134
+ let { functionSetting = [] } = serverSetting;
1135
+ list = functionSetting;
1136
+ } else {
1137
+ let curList = listClassify.find(item => item.classify == this.classify);
1138
+ if (!curList) return false;
1139
+ list = curList.serverFunctionSetting || [];
1140
+ }
1141
+ if (!list.length) return true;
1142
+ let curItem = list.find(item => item.function == 'ACCESS_INTERCEPT');
1143
+ if (!curItem) return true;
1144
+ return curItem.isChecked == 'Y';
1145
+ },
1146
+ accessChat(data) {
1147
+ let { bid: userId, username, userType } = this.queueItem;
1148
+ let params = {
1149
+ assemblyId: this.assemblyId,
1150
+ userId,
1151
+ userType,
1152
+ lastSessionId: this.sessionId,
1153
+ relateId: this.clientParams.relateId
1154
+ };
1155
+ if (data) {
1156
+ params.addParams = JSON.stringify(data);
1157
+ }
1158
+ return fetch.post('/chat/service/accessChat', qs.stringify(params)).then(({ data }) => {
1159
+ if (data.result === 'SUCCESS') {
1160
+ let { countdown, clientParams, sessionType, countdownDesc } = data.map;
1161
+ this.setSessionId(data.map.sessionId);
1162
+ this.clearMsgList();
1163
+ this.setMsgList();
1164
+ this.setCurrentTab('session');
1165
+ this.setLastCurrentTab('session');
1166
+ this.setClientId(username);
1167
+ this.setOnChating(true);
1168
+ this.setSessionEnd(false);
1169
+ this.setQueueItem(null);
1170
+ sessionType && this.setSessionType(sessionType);
1171
+ if (countdown >= 0) {
1172
+ this.setChatTimer({ countdown, countdownDesc });
1173
+ }
1174
+ this.setClientParams(clientParams);
1175
+ this.getSessionHistoryList();
1176
+ this.dispatchEvent('user_accessSuccess', {
1177
+ map: data.map,
1178
+ user: { ...this.curScrollItem }
1179
+ });
1180
+ } else {
1181
+ this.$message.warning(data.resultMsg);
1182
+ }
1183
+ });
1184
+ },
1185
+ handleOpenTransfer() {
1186
+ if (this.enableTransfer) {
1187
+ this.transferVisible = true;
1188
+ }
1189
+ },
1190
+ handleTransfer(data) {
1191
+ const receiveId = data.converUserId;
1192
+ if (receiveId) {
1193
+ if (this.userId == receiveId) {
1194
+ this.$message.warning(this.i18nText('1.2.7.14.1'));
1195
+ return;
1196
+ }
1197
+ this.transferVisible = false;
1198
+ fetch({
1199
+ url: '/chat/service/transferQueue',
1200
+ method: 'POST',
1201
+ data: qs.stringify({
1202
+ relateId: this.clientParams.relateId,
1203
+ orgId: this.orgId,
1204
+ assemblyId: this.assemblyId,
1205
+ userId: this.curScrollItem?.bid,
1206
+ userType: this.curScrollItem?.userType || '',
1207
+ receiveId: receiveId,
1208
+ deptId: data.deptId,
1209
+ remark: data.converMark
1210
+ })
1211
+ })
1212
+ .then(({ data }) => {
1213
+ if (data.result === 'SUCCESS') {
1214
+ this.dispatchEvent('user_transferQueueSuccess', {
1215
+ user: { ...this.curScrollItem }
1216
+ });
1217
+ this.$message.success(this.i18nText('1.9.336'));
1218
+ this.clearCurrentSession(true);
1219
+ } else {
1220
+ this.$message.warning(data.resultMsg || this.i18nText('1.10.201') + this.i18nText('1.9.22'));
1221
+ }
1222
+ })
1223
+ .catch(err => {
1224
+ this.$message.warning(err.resultMsg || this.i18nText('1.10.201') + this.i18nText('1.9.22'));
1225
+ });
1226
+ } else {
1227
+ this.$message.warning(this.i18nText('1.9.335'));
1228
+ }
1229
+ },
1230
+ getSessionHistoryList() {
1231
+ this.setSessionHistoryList([]);
1232
+ if (this.assemblySetting.recordScopeSetting === 'CURRENT') return;
1233
+ fetch
1234
+ .get('/chat/service/getPastSessionList', {
1235
+ params: { sessionId: this.sessionId }
1236
+ })
1237
+ .then(({ data }) => {
1238
+ if (data.result === 'SUCCESS') {
1239
+ let list = data.list.map(item => {
1240
+ if (item.id === this.sessionId || this.type === 'session') {
1241
+ item.onChating = true;
1242
+ }
1243
+ return item;
1244
+ });
1245
+ this.setSessionHistoryList(list);
1246
+ }
1247
+ });
1248
+ },
1249
+ openQuickData() {
1250
+ // this.quickShow = true;
1251
+ if (this.quickData.length < 1) {
1252
+ fetch
1253
+ .get('/fastReply/getChatList', {
1254
+ params: {
1255
+ category: this.assemblySetting.chatPublicCategory
1256
+ }
1257
+ })
1258
+ .then(({ data }) => {
1259
+ if (data.result === 'SUCCESS') {
1260
+ this.quickData = data.list;
1261
+ }
1262
+ });
1263
+ }
1264
+ },
1265
+ // 清除当前会话
1266
+ clearCurrentSession(update = false) {
1267
+ this.clearMsgList([]);
1268
+ this.setSessionHistoryList([]);
1269
+ this.setSessionTimer('');
1270
+ this.setClientId('');
1271
+ this.setSessionId('');
1272
+ this.setScrollQueueId('');
1273
+ this.setLastClassify('');
1274
+ this.setCurScrollItem(null);
1275
+ this.setQueueItem(null);
1276
+ this.setClientParams(null);
1277
+ if (update) {
1278
+ // 重新设置 currentTab 触发 scrollList 组件 更新排队列表
1279
+ const currentTab = this.store.state?.currentTab;
1280
+ if (currentTab === 'queue') {
1281
+ this.setCurrentTab('');
1282
+ this.$nextTick(() => {
1283
+ this.setCurrentTab(this.store.state?.currentTab || currentTab);
1284
+ });
1285
+ }
1286
+ }
1287
+ },
1288
+ openRefuseReason() {
1289
+ if (this.enableRefuse) {
1290
+ this.refuseVisible = true;
1291
+ }
1292
+ },
1293
+ handleRefuse(remark) {
1294
+ if (!this.enableRefuse) return;
1295
+ fetch({
1296
+ url: '/chat/service/accessRefuse',
1297
+ method: 'POST',
1298
+ data: qs.stringify({
1299
+ relateId: this.clientParams.relateId,
1300
+ orgId: this.orgId,
1301
+ assemblyId: this.assemblyId,
1302
+ userId: this.curScrollItem?.bid,
1303
+ userType: this.curScrollItem?.userType || '',
1304
+ remark: remark || '其他'
1305
+ })
1306
+ })
1307
+ .then(({ data }) => {
1308
+ if (data.result === 'SUCCESS') {
1309
+ this.dispatchEvent('user_refuseSuccess', {
1310
+ user: { ...this.curScrollItem }
1311
+ });
1312
+ this.$message.success(this.i18nText('1.10.184'));
1313
+ this.clearCurrentSession(true);
1314
+ } else {
1315
+ this.$message.warning(data.resultMsg || this.i18nText('1.9.353') + this.i18nText('1.9.22'));
1316
+ }
1317
+ })
1318
+ .catch(err => {
1319
+ this.$message.warning(err.resultMsg || this.i18nText('1.9.353') + this.i18nText('1.9.22'));
1320
+ });
1321
+ this.refuseVisible = false;
1322
+ },
1323
+ bindEvents() {
1324
+ const handleDrop = e => {
1325
+ e.stopPropagation();
1326
+ e.preventDefault();
1327
+ this.handleFiles(e.dataTransfer.items);
1328
+ };
1329
+ document.addEventListener('drop', handleDrop, false);
1330
+ this.registerEvent('accessCurrentSession', this.handleAccess);
1331
+ this.registerEvent('refuseCurrentSession', this.openRefuseReason);
1332
+ this.registerEvent('transferCurrentQueueSession', this.handleOpenTransfer);
1333
+ this.$on('hook:beforeDestroy', () => {
1334
+ document.removeEventListener('drop', handleDrop, false);
1335
+ window.removeEventListener('message', this.iframeEvent);
1336
+ window.removeEventListener('message', this.handleAccessIntercept);
1337
+ this.unregisterEvent('accessCurrentSession', this.handleAccess);
1338
+ this.unregisterEvent('refuseCurrentSession', this.openRefuseReason);
1339
+ this.unregisterEvent('transferCurrentQueueSession', this.handleOpenTransfer);
1340
+ });
1341
+ }
1342
+ },
1343
+ watch: {
1344
+ accessIntercept: {
1345
+ handler(val) {
1346
+ if (!val) {
1347
+ window.removeEventListener('message', this.handleAccessIntercept);
1348
+ }
1349
+ }
1350
+ },
1351
+ message: {
1352
+ handler(v) {
1353
+ // 使用 _editValue 判断是否组件外修改 message,并更新聊天框
1354
+ if (this._editValue !== v) {
1355
+ this.updateMessage(v);
1356
+ }
1357
+ }
1358
+ }
1359
+ }
1360
+ };
1361
+ </script>
1362
+
1363
+ <style lang="less">
1364
+ .chat-footer-modal-big {
1365
+ padding: 10px;
1366
+ &.ant-modal-wrap {
1367
+ overflow: none;
1368
+ }
1369
+ .ant-modal {
1370
+ width: 100%;
1371
+ height: 100%;
1372
+ top: 0;
1373
+ padding-bottom: 0;
1374
+ }
1375
+ .ant-modal-content {
1376
+ height: 100%;
1377
+ }
1378
+ .ant-modal-body {
1379
+ height: calc(100% - 55px);
1380
+ }
1381
+ }
1382
+ .chat-footer-modal-small {
1383
+ &.ant-modal-wrap {
1384
+ overflow: none;
1385
+ }
1386
+ .ant-modal {
1387
+ width: 80%;
1388
+ height: 80%;
1389
+ padding-bottom: 0;
1390
+ top: 60px;
1391
+ }
1392
+ .ant-modal-content {
1393
+ height: 100%;
1394
+ }
1395
+ .ant-modal-body {
1396
+ height: 100%;
1397
+ padding: 0;
1398
+ }
1399
+ }
1400
+
1401
+ </style>
1402
+ <style lang="less" scoped>
1403
+ .message-operate {
1404
+ display: flex;
1405
+ align-items: center;
1406
+ height: 40px;
1407
+ line-height: 18px;
1408
+ background: #fff;
1409
+ overflow-x: auto;
1410
+ overflow-y: hidden;
1411
+ border-top: 1px solid #e6e6e6;
1412
+ &::-webkit-scrollbar {
1413
+ height: 5px;
1414
+ }
1415
+ > img {
1416
+ // > .operate-icon {
1417
+ margin-left: 10px;
1418
+ }
1419
+ /deep/ .ant-tabs .ant-tabs-top-content > .ant-tabs-tabpane {
1420
+ height: 300px !important;
1421
+ overflow: auto !important;
1422
+ }
1423
+ /deep/ .ant-upload-list {
1424
+ display: none;
1425
+ }
1426
+ /deep/ .ant-upload {
1427
+ height: 32px;
1428
+ vertical-align: middle;
1429
+ display: inline-block;
1430
+ }
1431
+ .no-click {
1432
+ pointer-events: none;
1433
+ opacity: 0.4;
1434
+ }
1435
+ .svg-icon {
1436
+ color: rgba(0, 0, 0, 0.6);
1437
+ }
1438
+ }
1439
+ .emoji-wrapper {
1440
+ width: 500px;
1441
+ height: 300px;
1442
+ overflow: auto;
1443
+ display: grid;
1444
+ grid-template-columns: repeat(10, 1fr);
1445
+ .emoji {
1446
+ display: inline-block;
1447
+ font-size: 20px;
1448
+ margin: 10px;
1449
+ text-align: center;
1450
+ }
1451
+ &::-webkit-scrollbar {
1452
+ width: 5px;
1453
+ }
1454
+ }
1455
+ .message {
1456
+ position: relative;
1457
+ height: 150px;
1458
+ .message-send {
1459
+ position: absolute;
1460
+ bottom: 8px;
1461
+ right: 14px;
1462
+ height: 40px;
1463
+ border-radius: 20px;
1464
+ width: 86px;
1465
+ font-weight: 600;
1466
+ font-size: 18px;
1467
+ color: #fff;
1468
+ background-color: @primary-color;
1469
+ border-color: @primary-color;
1470
+ &:hover {
1471
+ background-color: rgba(@primary-color, 0.7);
1472
+ border-color: rgba(@primary-color, 0.7);
1473
+ }
1474
+ &[disabled]:hover,
1475
+ &[disabled] {
1476
+ background-color: #f5f5f5;
1477
+ border-color: #d9d9d9;
1478
+ color: #bebebe;
1479
+ }
1480
+ }
1481
+ .prompt {
1482
+ position: absolute;
1483
+ bottom: 16px;
1484
+ right: 120px;
1485
+ color: #a6a6a6;
1486
+ }
1487
+ }
1488
+ .infinite-container {
1489
+ overflow-y: auto;
1490
+ }
1491
+ .message-wrapper {
1492
+ position: relative;
1493
+ .loading-icon {
1494
+ position: absolute;
1495
+ top: 0;
1496
+ left: 50%;
1497
+ }
1498
+ }
1499
+ .content-edit {
1500
+ width: 100%;
1501
+ height: 100%;
1502
+ padding: 0 16px 56px;
1503
+ word-break: break-word;
1504
+ &::-webkit-scrollbar {
1505
+ width: 5px;
1506
+ }
1507
+ cursor: text;
1508
+ > div {
1509
+ height: 100%;
1510
+ outline: none;
1511
+ overflow-y: auto;
1512
+ overflow-x: hidden;
1513
+ }
1514
+ }
1515
+ .overlay {
1516
+ position: absolute;
1517
+ width: 100%;
1518
+ height: 100%;
1519
+ top: 0;
1520
+ cursor: not-allowed;
1521
+ }
1522
+ .queue-page {
1523
+ display: flex;
1524
+ flex-direction: column;
1525
+ align-items: center;
1526
+ justify-content: center;
1527
+ position: absolute;
1528
+ width: 100%;
1529
+ height: 100%;
1530
+ top: 0;
1531
+ background: #fff;
1532
+ z-index: 500;
1533
+ border-top: 1px solid #e6e6e6;
1534
+ .queue-tips {
1535
+ font-size: 16px;
1536
+ color: #272727;
1537
+ line-height: 22px;
1538
+ margin-bottom: 16px;
1539
+ }
1540
+ .queue-btns {
1541
+ display: flex;
1542
+ justify-content: center;
1543
+ }
1544
+ .toSession {
1545
+ width: 80px;
1546
+ height: 80px;
1547
+ line-height: 80px;
1548
+ text-align: center;
1549
+ background: linear-gradient(180deg, #ffffff 0%, #f5f5f5 100%);
1550
+ box-shadow: 0px 4px 9px 0px rgba(186, 186, 186, 0.79);
1551
+ border-radius: 50%;
1552
+ color: #4b4b4b;
1553
+ font-size: 20px;
1554
+ font-weight: 500;
1555
+ font-family: PingFangSC-Medium, PingFang SC;
1556
+ transition: transform 0.3s ease;
1557
+ &.primary {
1558
+ background: linear-gradient(180deg, #2d7aff 0%, #2c5df4 100%);
1559
+ box-shadow: 0px 5px 8px 0px rgba(44, 101, 247, 0.39);
1560
+ color: #fff;
1561
+ }
1562
+ &.danger {
1563
+ background: linear-gradient(180deg, #ff4b4b 0%, #c80606 100%);
1564
+ box-shadow: 0px 5px 8px 0px rgba(217, 26, 26, 0.35);
1565
+ color: #fff;
1566
+ }
1567
+ &:hover {
1568
+ transform: scale(1.2);
1569
+ }
1570
+ & + .toSession {
1571
+ margin-left: 36px;
1572
+ }
1573
+ }
1574
+ }
1575
+ .toolbar {
1576
+ // margin-left: 10px;
1577
+ margin: 0 5px;
1578
+ font-size: 20px;
1579
+ outline: none;
1580
+ padding: 6px;
1581
+ box-sizing: content-box;
1582
+ &:hover {
1583
+ background-color: #ebebeb;
1584
+ border-radius: 6px;
1585
+ }
1586
+ }
1587
+ .closed-toolbar {
1588
+ z-index: 100;
1589
+ }
1590
+ /deep/ p {
1591
+ margin-bottom: 0;
1592
+ }
1593
+ .click-voice {
1594
+ color: #2d7aff;
1595
+ }
1596
+ </style>