@tencentcloud/ai-desk-customer-vue 0.1.0

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 (351) hide show
  1. package/.code.yml +17 -0
  2. package/.eslintignore +6 -0
  3. package/.stylelintrc.json +40 -0
  4. package/CHANGELOG.md +5 -0
  5. package/README.md +12 -0
  6. package/adapter-vue-web.ts +73 -0
  7. package/adapter-vue.ts +10 -0
  8. package/assets/audio-before-delete.svg +4 -0
  9. package/assets/audio-blue.png +0 -0
  10. package/assets/audio-bubble-blue.svg +3 -0
  11. package/assets/audio-bubble-red.svg +3 -0
  12. package/assets/audio-delete.svg +10 -0
  13. package/assets/audio.svg +6 -0
  14. package/assets/back.svg +16 -0
  15. package/assets/background_mobile.png +0 -0
  16. package/assets/camera-uni.png +0 -0
  17. package/assets/close-quote-icon.svg +3 -0
  18. package/assets/customer_avatar.png +0 -0
  19. package/assets/customer_avatar_mobile.png +0 -0
  20. package/assets/dialog-close.png +0 -0
  21. package/assets/double-arrow.svg +1 -0
  22. package/assets/download.svg +6 -0
  23. package/assets/emoji.png +0 -0
  24. package/assets/face-uni.png +0 -0
  25. package/assets/face.png +0 -0
  26. package/assets/file-h5.png +0 -0
  27. package/assets/files.png +0 -0
  28. package/assets/form-dialog-bg.png +0 -0
  29. package/assets/icon-arrow-left.svg +7 -0
  30. package/assets/icon-close.svg +6 -0
  31. package/assets/iconRight.svg +3 -0
  32. package/assets/icon_form.png +0 -0
  33. package/assets/icon_form_filled.png +0 -0
  34. package/assets/icon_question.png +0 -0
  35. package/assets/icon_refresh.png +0 -0
  36. package/assets/icon_success.png +0 -0
  37. package/assets/imRobotGuess.svg +4 -0
  38. package/assets/image-uni.png +0 -0
  39. package/assets/image.png +0 -0
  40. package/assets/keyboard_icon.png +0 -0
  41. package/assets/loading.png +0 -0
  42. package/assets/more-uni.png +0 -0
  43. package/assets/more_tools.png +0 -0
  44. package/assets/msg-audio.svg +1 -0
  45. package/assets/msg-copy.svg +30 -0
  46. package/assets/msg-del.svg +33 -0
  47. package/assets/msg-quote.svg +8 -0
  48. package/assets/msg-revoke.svg +29 -0
  49. package/assets/radio-check.png +0 -0
  50. package/assets/radio-uncheck.png +0 -0
  51. package/assets/radio.svg +6 -0
  52. package/assets/refresh.svg +4 -0
  53. package/assets/rotate-left.svg +7 -0
  54. package/assets/rotate-right.svg +7 -0
  55. package/assets/star.png +0 -0
  56. package/assets/starLine.png +0 -0
  57. package/assets/translate.svg +12 -0
  58. package/assets/video-play.png +0 -0
  59. package/assets/video-uni.png +0 -0
  60. package/assets/video.png +0 -0
  61. package/assets/zoom-in.svg +9 -0
  62. package/assets/zoom-out.svg +9 -0
  63. package/components/CustomerServiceChat/chat-header/index-web.vue +141 -0
  64. package/components/CustomerServiceChat/emoji-config/custom-emoji.ts +15 -0
  65. package/components/CustomerServiceChat/emoji-config/default-emoji.ts +114 -0
  66. package/components/CustomerServiceChat/emoji-config/index.ts +140 -0
  67. package/components/CustomerServiceChat/emoji-config/locales/en.ts +66 -0
  68. package/components/CustomerServiceChat/emoji-config/locales/zh_cn.ts +66 -0
  69. package/components/CustomerServiceChat/index-web.vue +250 -0
  70. package/components/CustomerServiceChat/message-input/index-web.vue +214 -0
  71. package/components/CustomerServiceChat/message-input/index.ts +2 -0
  72. package/components/CustomerServiceChat/message-input/message-input-button.vue +95 -0
  73. package/components/CustomerServiceChat/message-input/message-input-editor-web.vue +884 -0
  74. package/components/CustomerServiceChat/message-input/message-input-file-web.ts +38 -0
  75. package/components/CustomerServiceChat/message-input/message-input-quote/index.vue +204 -0
  76. package/components/CustomerServiceChat/message-input-toolbar/emoji-dialog-mobile/emoji-dialog-mobile.vue +145 -0
  77. package/components/CustomerServiceChat/message-input-toolbar/emoji-picker/emoji-picker-dialog.vue +180 -0
  78. package/components/CustomerServiceChat/message-input-toolbar/emoji-picker/index.ts +2 -0
  79. package/components/CustomerServiceChat/message-input-toolbar/emoji-picker/index.vue +90 -0
  80. package/components/CustomerServiceChat/message-input-toolbar/emoji-picker/style/h5.scss +26 -0
  81. package/components/CustomerServiceChat/message-input-toolbar/emoji-picker/style/index.scss +4 -0
  82. package/components/CustomerServiceChat/message-input-toolbar/emoji-picker/style/web.scss +55 -0
  83. package/components/CustomerServiceChat/message-input-toolbar/file-upload/index.ts +2 -0
  84. package/components/CustomerServiceChat/message-input-toolbar/file-upload/index.vue +77 -0
  85. package/components/CustomerServiceChat/message-input-toolbar/image-upload/index.ts +2 -0
  86. package/components/CustomerServiceChat/message-input-toolbar/image-upload/index.vue +185 -0
  87. package/components/CustomerServiceChat/message-input-toolbar/index-web.vue +191 -0
  88. package/components/CustomerServiceChat/message-input-toolbar/index.ts +2 -0
  89. package/components/CustomerServiceChat/message-input-toolbar/style/uni.scss +111 -0
  90. package/components/CustomerServiceChat/message-input-toolbar/toolbar-item-container/index.vue +149 -0
  91. package/components/CustomerServiceChat/message-input-toolbar/toolbar-item-container/style/color.scss +6 -0
  92. package/components/CustomerServiceChat/message-input-toolbar/toolbar-item-container/style/h5.scss +20 -0
  93. package/components/CustomerServiceChat/message-input-toolbar/toolbar-item-container/style/index.scss +5 -0
  94. package/components/CustomerServiceChat/message-input-toolbar/toolbar-item-container/style/uni.scss +36 -0
  95. package/components/CustomerServiceChat/message-input-toolbar/toolbar-item-container/style/web.scss +19 -0
  96. package/components/CustomerServiceChat/message-input-toolbar/video-upload/index.ts +2 -0
  97. package/components/CustomerServiceChat/message-input-toolbar/video-upload/index.vue +146 -0
  98. package/components/CustomerServiceChat/message-list/index-web.vue +712 -0
  99. package/components/CustomerServiceChat/message-list/link/index.ts +23 -0
  100. package/components/CustomerServiceChat/message-list/message-elements/message-audio-web.vue +223 -0
  101. package/components/CustomerServiceChat/message-list/message-elements/message-bubble-web.vue +504 -0
  102. package/components/CustomerServiceChat/message-list/message-elements/message-custom.vue +5 -0
  103. package/components/CustomerServiceChat/message-list/message-elements/message-face.vue +45 -0
  104. package/components/CustomerServiceChat/message-list/message-elements/message-file.vue +151 -0
  105. package/components/CustomerServiceChat/message-list/message-elements/message-image-web.vue +97 -0
  106. package/components/CustomerServiceChat/message-list/message-elements/message-location.vue +34 -0
  107. package/components/CustomerServiceChat/message-list/message-elements/message-quote/index-web.vue +286 -0
  108. package/components/CustomerServiceChat/message-list/message-elements/message-quote/interface.ts +60 -0
  109. package/components/CustomerServiceChat/message-list/message-elements/message-record/index.vue +139 -0
  110. package/components/CustomerServiceChat/message-list/message-elements/message-text.vue +94 -0
  111. package/components/CustomerServiceChat/message-list/message-elements/message-timestamp.vue +76 -0
  112. package/components/CustomerServiceChat/message-list/message-elements/message-video-web.vue +319 -0
  113. package/components/CustomerServiceChat/message-list/message-elements/plugins/plugin-components/index.ts +9 -0
  114. package/components/CustomerServiceChat/message-list/message-elements/plugins/plugin-components/message-customer/index.ts +5 -0
  115. package/components/CustomerServiceChat/message-list/message-elements/plugins/plugin-components/message-customer/message-customer-service.vue +22 -0
  116. package/components/CustomerServiceChat/message-list/message-elements/plugins/plugin-components/message-plugin-layout-web.vue +133 -0
  117. package/components/CustomerServiceChat/message-list/message-elements/plugins/plugin-components/message-plugin-web.vue +83 -0
  118. package/components/CustomerServiceChat/message-list/message-elements/read-status/index.vue +193 -0
  119. package/components/CustomerServiceChat/message-list/message-elements/simple-message-list/index.vue +462 -0
  120. package/components/CustomerServiceChat/message-list/message-elements/simple-message-list/message-container.vue +105 -0
  121. package/components/CustomerServiceChat/message-list/message-elements/video-play.vue +59 -0
  122. package/components/CustomerServiceChat/message-list/message-tool/index-web.vue +312 -0
  123. package/components/CustomerServiceChat/message-list/message-tool/message-revoked.vue +63 -0
  124. package/components/CustomerServiceChat/message-list/scroll-button/index.vue +201 -0
  125. package/components/CustomerServiceChat/message-list/style/color.scss +32 -0
  126. package/components/CustomerServiceChat/message-list/style/h5.scss +16 -0
  127. package/components/CustomerServiceChat/message-list/style/index.scss +11 -0
  128. package/components/CustomerServiceChat/message-list/style/web.scss +180 -0
  129. package/components/CustomerServiceChat/style/common.scss +59 -0
  130. package/components/CustomerServiceChat/style/h5.scss +51 -0
  131. package/components/CustomerServiceChat/style/index.scss +12 -0
  132. package/components/CustomerServiceChat/style/uni.scss +13 -0
  133. package/components/CustomerServiceChat/style/web.scss +46 -0
  134. package/components/CustomerServiceChat/style/wx.scss +5 -0
  135. package/components/CustomerServiceChat/utils/conversationDraft.ts +86 -0
  136. package/components/CustomerServiceChat/utils/sendMessage.ts +140 -0
  137. package/components/common/Avatar/index.vue +148 -0
  138. package/components/common/BottomPopup/index.ts +3 -0
  139. package/components/common/BottomPopup/index.vue +160 -0
  140. package/components/common/BottomPopup/style/h5.scss +62 -0
  141. package/components/common/BottomPopup/style/index.scss +3 -0
  142. package/components/common/BottomPopup/style/modal.scss +5 -0
  143. package/components/common/Dialog/index.ts +3 -0
  144. package/components/common/Dialog/index.vue +120 -0
  145. package/components/common/Dialog/style/color.scss +43 -0
  146. package/components/common/Dialog/style/dialog.scss +4 -0
  147. package/components/common/Dialog/style/h5.scss +56 -0
  148. package/components/common/Dialog/style/web.scss +61 -0
  149. package/components/common/Drawer/index.vue +164 -0
  150. package/components/common/Icon.vue +83 -0
  151. package/components/common/ImagePreviewer/image-item-web.vue +42 -0
  152. package/components/common/ImagePreviewer/index-web.vue +682 -0
  153. package/components/common/ImagePreviewer/index.ts +3 -0
  154. package/components/common/Overlay/index.vue +126 -0
  155. package/components/common/ProgressMessage/index.vue +95 -0
  156. package/components/common/RadioSelect/index.vue +59 -0
  157. package/components/common/RichText/LICENSE +21 -0
  158. package/components/common/RichText/README.md +244 -0
  159. package/components/common/RichText/dist/mp-alipay/index.acss +1 -0
  160. package/components/common/RichText/dist/mp-alipay/index.axml +1 -0
  161. package/components/common/RichText/dist/mp-alipay/index.js +8 -0
  162. package/components/common/RichText/dist/mp-alipay/index.json +1 -0
  163. package/components/common/RichText/dist/mp-alipay/node/node.acss +1 -0
  164. package/components/common/RichText/dist/mp-alipay/node/node.axml +1 -0
  165. package/components/common/RichText/dist/mp-alipay/node/node.js +1 -0
  166. package/components/common/RichText/dist/mp-alipay/node/node.json +1 -0
  167. package/components/common/RichText/dist/mp-alipay/parser.js +1 -0
  168. package/components/common/RichText/dist/mp-baidu/index.css +1 -0
  169. package/components/common/RichText/dist/mp-baidu/index.js +8 -0
  170. package/components/common/RichText/dist/mp-baidu/index.json +1 -0
  171. package/components/common/RichText/dist/mp-baidu/index.swan +1 -0
  172. package/components/common/RichText/dist/mp-baidu/node/node.css +1 -0
  173. package/components/common/RichText/dist/mp-baidu/node/node.js +1 -0
  174. package/components/common/RichText/dist/mp-baidu/node/node.json +1 -0
  175. package/components/common/RichText/dist/mp-baidu/node/node.swan +1 -0
  176. package/components/common/RichText/dist/mp-baidu/parser.js +1 -0
  177. package/components/common/RichText/dist/mp-qq/index.js +8 -0
  178. package/components/common/RichText/dist/mp-qq/index.json +1 -0
  179. package/components/common/RichText/dist/mp-qq/index.qml +1 -0
  180. package/components/common/RichText/dist/mp-qq/index.qss +1 -0
  181. package/components/common/RichText/dist/mp-qq/node/node.js +1 -0
  182. package/components/common/RichText/dist/mp-qq/node/node.json +1 -0
  183. package/components/common/RichText/dist/mp-qq/node/node.qml +1 -0
  184. package/components/common/RichText/dist/mp-qq/node/node.qss +1 -0
  185. package/components/common/RichText/dist/mp-qq/parser.js +1 -0
  186. package/components/common/RichText/dist/mp-toutiao/index.js +8 -0
  187. package/components/common/RichText/dist/mp-toutiao/index.json +1 -0
  188. package/components/common/RichText/dist/mp-toutiao/index.ttml +1 -0
  189. package/components/common/RichText/dist/mp-toutiao/index.ttss +1 -0
  190. package/components/common/RichText/dist/mp-toutiao/node/node.js +1 -0
  191. package/components/common/RichText/dist/mp-toutiao/node/node.json +1 -0
  192. package/components/common/RichText/dist/mp-toutiao/node/node.ttml +1 -0
  193. package/components/common/RichText/dist/mp-toutiao/node/node.ttss +1 -0
  194. package/components/common/RichText/dist/mp-toutiao/parser.js +1 -0
  195. package/components/common/RichText/dist/mp-weixin/index.js +8 -0
  196. package/components/common/RichText/dist/mp-weixin/index.json +1 -0
  197. package/components/common/RichText/dist/mp-weixin/index.wxml +1 -0
  198. package/components/common/RichText/dist/mp-weixin/index.wxss +1 -0
  199. package/components/common/RichText/dist/mp-weixin/node/node.js +1 -0
  200. package/components/common/RichText/dist/mp-weixin/node/node.json +1 -0
  201. package/components/common/RichText/dist/mp-weixin/node/node.wxml +1 -0
  202. package/components/common/RichText/dist/mp-weixin/node/node.wxss +1 -0
  203. package/components/common/RichText/dist/mp-weixin/parser.js +1 -0
  204. package/components/common/RichText/dist/uni-app/components/mp-html/mp-html.vue +498 -0
  205. package/components/common/RichText/dist/uni-app/components/mp-html/node/node.vue +587 -0
  206. package/components/common/RichText/dist/uni-app/components/mp-html/parser.js +1393 -0
  207. package/components/common/RichText/dist/uni-app/static/app-plus/mp-html/js/handler.js +1 -0
  208. package/components/common/RichText/dist/uni-app/static/app-plus/mp-html/js/uni.webview.min.js +1 -0
  209. package/components/common/RichText/dist/uni-app/static/app-plus/mp-html/local.html +1 -0
  210. package/components/common/RichText/docs/.nojekyll +0 -0
  211. package/components/common/RichText/docs/index.html +50 -0
  212. package/components/common/RichText/docs/lib/docsify.min.js +1 -0
  213. package/components/common/RichText/docs/lib/prism-bash.min.js +1 -0
  214. package/components/common/RichText/docs/lib/search.min.js +1 -0
  215. package/components/common/RichText/docs/lib/vue.css +858 -0
  216. package/components/common/RichText/gulpfile.js +113 -0
  217. package/components/common/RichText/lint.js +63 -0
  218. package/components/common/RichText/package.json +74 -0
  219. package/components/common/RichText/plugins/README.md +2 -0
  220. package/components/common/RichText/plugins/audio/README.md +25 -0
  221. package/components/common/RichText/plugins/audio/build.js +11 -0
  222. package/components/common/RichText/plugins/audio/context.js +7 -0
  223. package/components/common/RichText/plugins/audio/index.js +34 -0
  224. package/components/common/RichText/plugins/audio/miniprogram/audio.js +189 -0
  225. package/components/common/RichText/plugins/audio/miniprogram/audio.json +3 -0
  226. package/components/common/RichText/plugins/audio/miniprogram/audio.wxml +17 -0
  227. package/components/common/RichText/plugins/audio/miniprogram/audio.wxss +127 -0
  228. package/components/common/RichText/plugins/audio/miniprogram/build.js +3 -0
  229. package/components/common/RichText/plugins/audio/uni-app/audio.vue +269 -0
  230. package/components/common/RichText/plugins/audio/uni-app/build.js +3 -0
  231. package/components/common/RichText/plugins/card/README.md +30 -0
  232. package/components/common/RichText/plugins/card/build.js +14 -0
  233. package/components/common/RichText/plugins/card/index.js +7 -0
  234. package/components/common/RichText/plugins/card/miniprogram/build.js +3 -0
  235. package/components/common/RichText/plugins/card/miniprogram/card.js +26 -0
  236. package/components/common/RichText/plugins/card/miniprogram/card.json +3 -0
  237. package/components/common/RichText/plugins/card/miniprogram/card.wxml +9 -0
  238. package/components/common/RichText/plugins/card/miniprogram/card.wxss +55 -0
  239. package/components/common/RichText/plugins/card/uni-app/build.js +3 -0
  240. package/components/common/RichText/plugins/card/uni-app/card.vue +122 -0
  241. package/components/common/RichText/plugins/editable/README.md +137 -0
  242. package/components/common/RichText/plugins/editable/config.js +15 -0
  243. package/components/common/RichText/plugins/editable/miniprogram/build.js +813 -0
  244. package/components/common/RichText/plugins/editable/miniprogram/index.js +551 -0
  245. package/components/common/RichText/plugins/editable/uni-app/build.js +744 -0
  246. package/components/common/RichText/plugins/editable/uni-app/index.js +553 -0
  247. package/components/common/RichText/plugins/emoji/README.md +15 -0
  248. package/components/common/RichText/plugins/emoji/index.js +203 -0
  249. package/components/common/RichText/plugins/highlight/README.md +26 -0
  250. package/components/common/RichText/plugins/highlight/config.js +5 -0
  251. package/components/common/RichText/plugins/highlight/index.js +96 -0
  252. package/components/common/RichText/plugins/highlight/miniprogram/build.js +88 -0
  253. package/components/common/RichText/plugins/highlight/prism.css +125 -0
  254. package/components/common/RichText/plugins/highlight/prism.min.js +7 -0
  255. package/components/common/RichText/plugins/highlight/uni-app/build.js +88 -0
  256. package/components/common/RichText/plugins/img-cache/README.md +24 -0
  257. package/components/common/RichText/plugins/img-cache/build.js +16 -0
  258. package/components/common/RichText/plugins/img-cache/index.js +138 -0
  259. package/components/common/RichText/plugins/latex/README.md +16 -0
  260. package/components/common/RichText/plugins/latex/build.js +14 -0
  261. package/components/common/RichText/plugins/latex/index.js +77 -0
  262. package/components/common/RichText/plugins/latex/katex.css +1070 -0
  263. package/components/common/RichText/plugins/latex/katex.min.js +1 -0
  264. package/components/common/RichText/plugins/markdown/README.md +17 -0
  265. package/components/common/RichText/plugins/markdown/index.js +34 -0
  266. package/components/common/RichText/plugins/markdown/marked.min.js +6 -0
  267. package/components/common/RichText/plugins/markdown/miniprogram/build.js +70 -0
  268. package/components/common/RichText/plugins/markdown/uni-app/build.js +68 -0
  269. package/components/common/RichText/plugins/search/README.md +46 -0
  270. package/components/common/RichText/plugins/search/miniprogram/index.js +137 -0
  271. package/components/common/RichText/plugins/search/uni-app/index.js +132 -0
  272. package/components/common/RichText/plugins/style/README.md +30 -0
  273. package/components/common/RichText/plugins/style/index.js +129 -0
  274. package/components/common/RichText/plugins/style/parser.js +175 -0
  275. package/components/common/RichText/plugins/template/README.md +2 -0
  276. package/components/common/RichText/plugins/template/build.js +65 -0
  277. package/components/common/RichText/plugins/template/index.js +67 -0
  278. package/components/common/RichText/plugins/txv-video/README.md +18 -0
  279. package/components/common/RichText/plugins/txv-video/build.js +3 -0
  280. package/components/common/RichText/plugins/txv-video/index.js +46 -0
  281. package/components/common/RichText/plugins/txv-video/miniprogram/build.js +6 -0
  282. package/components/common/RichText/plugins/txv-video/uni-app/build.js +3 -0
  283. package/components/common/RichText/src/miniprogram/index.js +396 -0
  284. package/components/common/RichText/src/miniprogram/index.json +6 -0
  285. package/components/common/RichText/src/miniprogram/index.wxml +7 -0
  286. package/components/common/RichText/src/miniprogram/index.wxss +13 -0
  287. package/components/common/RichText/src/miniprogram/node/node.js +247 -0
  288. package/components/common/RichText/src/miniprogram/node/node.json +6 -0
  289. package/components/common/RichText/src/miniprogram/node/node.wxml +112 -0
  290. package/components/common/RichText/src/miniprogram/node/node.wxss +164 -0
  291. package/components/common/RichText/src/miniprogram/parser.js +1269 -0
  292. package/components/common/RichText/src/uni-app/components/mp-html/mp-html.vue +498 -0
  293. package/components/common/RichText/src/uni-app/components/mp-html/node/node.vue +585 -0
  294. package/components/common/RichText/src/uni-app/components/mp-html/parser.js +1393 -0
  295. package/components/common/RichText/src/uni-app/static/app-plus/mp-html/js/handler.js +254 -0
  296. package/components/common/RichText/src/uni-app/static/app-plus/mp-html/js/uni.webview.min.js +1 -0
  297. package/components/common/RichText/src/uni-app/static/app-plus/mp-html/local.html +33 -0
  298. package/components/common/RichText/tools/config.js +82 -0
  299. package/components/common/RichText/tools/converter.js +205 -0
  300. package/components/common/RichText/tools/ifdef.js +115 -0
  301. package/components/common/RichText/tools/minifier.js +40 -0
  302. package/components/common/RichText/tools/plugin.js +248 -0
  303. package/components/common/Toast/index-web.ts +122 -0
  304. package/components/common/Toast/index-web.vue +178 -0
  305. package/components/common/Toast/type.ts +8 -0
  306. package/components/common/common.scss +59 -0
  307. package/components/customer-icon.vue +56 -0
  308. package/components/message-branch.vue +120 -0
  309. package/components/message-customer-service.vue +114 -0
  310. package/components/message-form/form-branch.vue +68 -0
  311. package/components/message-form/form-input.vue +242 -0
  312. package/components/message-form/index.vue +80 -0
  313. package/components/message-multi-branch/branch-pc.vue +79 -0
  314. package/components/message-multi-branch/index.vue +60 -0
  315. package/components/message-multi-form/component-mobile/form-popup.vue +37 -0
  316. package/components/message-multi-form/component-mobile/input-mobile.vue +102 -0
  317. package/components/message-multi-form/component-mobile/label-mobile.vue +32 -0
  318. package/components/message-multi-form/component-mobile/radios-mobile.vue +161 -0
  319. package/components/message-multi-form/component-pc/input-pc.vue +93 -0
  320. package/components/message-multi-form/component-pc/label-pc.vue +33 -0
  321. package/components/message-multi-form/component-pc/radio-pc.vue +98 -0
  322. package/components/message-multi-form/form-mobile.vue +385 -0
  323. package/components/message-multi-form/form-pc.vue +184 -0
  324. package/components/message-multi-form/index.vue +61 -0
  325. package/components/message-product-card.vue +129 -0
  326. package/components/message-rating/index.vue +56 -0
  327. package/components/message-rating/message-rating-number.vue +247 -0
  328. package/components/message-rating/message-rating-star.vue +237 -0
  329. package/components/message-rich-text.vue +155 -0
  330. package/components/message-robot-welcome.vue +181 -0
  331. package/components/message-stream.vue +109 -0
  332. package/constant.ts +122 -0
  333. package/excluded-list.txt +6 -0
  334. package/index.ts +30 -0
  335. package/index.vue +60 -0
  336. package/interface.ts +176 -0
  337. package/logger/index.ts +12 -0
  338. package/logger/main.ts +120 -0
  339. package/package.json +62 -0
  340. package/script/fileCopy.js +60 -0
  341. package/script/syncVersion.js +35 -0
  342. package/server.ts +128 -0
  343. package/styles/common.scss +116 -0
  344. package/tsconfig.json +34 -0
  345. package/typings.d.ts +20 -0
  346. package/utils/chatStorage.ts +70 -0
  347. package/utils/copy-web.ts +141 -0
  348. package/utils/enableSampleTaskStatus.ts +8 -0
  349. package/utils/env.ts +15 -0
  350. package/utils/index.ts +59 -0
  351. package/utils/utils.ts +162 -0
@@ -0,0 +1,712 @@
1
+ <template>
2
+ <div
3
+ class="tui-chat"
4
+ :class="[isH5 ? 'tui-chat-h5' : '']"
5
+ >
6
+ <div
7
+ id="tui-chat-main"
8
+ class="tui-chat-main"
9
+ @click="closeChatPop"
10
+ >
11
+ <!-- Message List -->
12
+ <ul
13
+ id="messageScrollList"
14
+ ref="messageListRef"
15
+ class="tui-message-list"
16
+ @click="onMessageListBackgroundClick"
17
+ >
18
+ <p
19
+ v-if="!isCompleted"
20
+ class="message-more"
21
+ @click="getHistoryMessageList"
22
+ >
23
+ {{ TUITranslateService.t("TUIChat.查看更多") }}
24
+ </p>
25
+ <li
26
+ v-for="(item, index) in messageList"
27
+ :id="'tui-' + item.ID"
28
+ :key="item.ID"
29
+ ref="messageElementListRef"
30
+ class="message-li"
31
+ >
32
+ <MessageTimestamp
33
+ :currTime="item.time"
34
+ :prevTime="index > 0 ? messageList[index - 1].time : 0"
35
+ />
36
+ <div class="message-item">
37
+ <MessageRevoked
38
+ v-if="item.isRevoked"
39
+ :isEdit="item.type === TYPES.MSG_TEXT"
40
+ :messageItem="item"
41
+ @messageEdit="handleEdit(item)"
42
+ />
43
+ <MessagePlugin
44
+ v-else-if="isPluginMessage(item)"
45
+ :message="deepCopy(item)"
46
+ :blinkMessageIDList="blinkMessageIDList"
47
+ @resendMessage="resendMessage"
48
+ @handleToggleMessageItem="handleToggleMessageItem"
49
+ @handleH5LongPress="handleH5LongPress"
50
+ />
51
+ <div
52
+ v-else
53
+ :class="{
54
+ 'message-event-bind-div': true,
55
+ }"
56
+ @longpress="handleToggleMessageItem($event, item, true)"
57
+ @click.prevent.right="handleToggleMessageItemForPC($event, item)"
58
+ @touchstart="handleH5LongPress($event, item, 'touchstart')"
59
+ @touchend="handleH5LongPress($event, item, 'touchend')"
60
+ @mouseover="handleH5LongPress($event, item, 'touchend')"
61
+ >
62
+ <MessageBubble
63
+ :content="item.getMessageContent()"
64
+ :isAudioPlayed="Boolean(audioPlayedMapping[item.ID])"
65
+ :blinkMessageIDList="blinkMessageIDList"
66
+ :messageItem="JSON.parse(JSON.stringify(item))"
67
+ @blinkMessage="blinkMessage"
68
+ @resendMessage="resendMessage(item)"
69
+ >
70
+ <template #messageElement>
71
+ <MessageText
72
+ v-if="item.type === TYPES.MSG_TEXT"
73
+ :content="item.getMessageContent()"
74
+ />
75
+ <ProgressMessage
76
+ v-else-if="item.type === TYPES.MSG_IMAGE"
77
+ :content="item.getMessageContent()"
78
+ :messageItem="item"
79
+ >
80
+ <MessageImage
81
+ :content="item.getMessageContent()"
82
+ :messageItem="item"
83
+ @previewImage="handleImagePreview"
84
+ />
85
+ </ProgressMessage>
86
+ <ProgressMessage
87
+ v-else-if="item.type === TYPES.MSG_VIDEO"
88
+ :content="item.getMessageContent()"
89
+ :messageItem="item"
90
+ >
91
+ <MessageVideo
92
+ :content="item.getMessageContent()"
93
+ :messageItem="item"
94
+ />
95
+ </ProgressMessage>
96
+ <MessageAudio
97
+ v-else-if="item.type === TYPES.MSG_AUDIO"
98
+ :content="item.getMessageContent()"
99
+ :messageItem="item"
100
+ @setAudioPlayed="setAudioPlayed"
101
+ />
102
+ <ProgressMessage
103
+ v-else-if="item.type === TYPES.MSG_FILE"
104
+ :content="item.getMessageContent()"
105
+ :messageItem="item"
106
+ >
107
+ <MessageFile
108
+ :content="item.getMessageContent()"
109
+ :messageItem="item"
110
+ />
111
+ </ProgressMessage>
112
+ <MessageRecord
113
+ v-else-if="item.type === TYPES.MSG_MERGER"
114
+ :renderData="item.payload"
115
+ :messageItem="item"
116
+ />
117
+ <MessageFace
118
+ v-else-if="item.type === TYPES.MSG_FACE"
119
+ :content="item.getMessageContent()"
120
+ />
121
+ <MessageLocation
122
+ v-else-if="item.type === TYPES.MSG_LOCATION"
123
+ :content="item.getMessageContent()"
124
+ />
125
+ <MessageCustom
126
+ v-else-if="item.type === TYPES.MSG_CUSTOM"
127
+ :content="item.getMessageContent()"
128
+ :messageItem="item"
129
+ />
130
+ </template>
131
+ </MessageBubble>
132
+ </div>
133
+ <!-- message tool -->
134
+ <MessageTool
135
+ v-if="item.ID === toggleID"
136
+ ref="messageToolListRef"
137
+ :class="{
138
+ 'message-tool': true,
139
+ 'message-tool-out': item.flow === 'out',
140
+ 'message-tool-in': item.flow === 'in',
141
+ 'message-tool-bottom': isTopMessageDom,
142
+ }"
143
+ :messageItem="item"
144
+ />
145
+ </div>
146
+ </li>
147
+ </ul>
148
+ <ScrollButton
149
+ ref="scrollButtonInstanceRef"
150
+ @scrollToLatestMessage="scrollToLatestMessage"
151
+ />
152
+ <Dialog
153
+ v-if="reSendDialogShow"
154
+ class="resend-dialog"
155
+ :show="reSendDialogShow"
156
+ :isH5="!isPC"
157
+ :center="true"
158
+ :isHeaderShow="isPC"
159
+ @submit="resendMessageConfirm()"
160
+ @update:show="(e) => (reSendDialogShow = e)"
161
+ >
162
+ <p class="delDialog-title">
163
+ {{ TUITranslateService.t("TUIChat.确认重发该消息?") }}
164
+ </p>
165
+ </Dialog>
166
+ <ImagePreviewer
167
+ v-if="showImagePreview"
168
+ :currentImage="currentImagePreview"
169
+ :imageList="imageMessageList"
170
+ @close="onImagePreviewerClose"
171
+ />
172
+ </div>
173
+ </div>
174
+ </template>
175
+
176
+ <script lang="ts" setup>
177
+ import vue from '../../../adapter-vue';
178
+ const { ref, nextTick, computed, onMounted, onUnmounted } = vue;
179
+ import TUIChatEngine, {
180
+ IMessageModel,
181
+ TUIStore,
182
+ StoreName,
183
+ TUITranslateService,
184
+ TUIChatService,
185
+ } from '@tencentcloud/chat-uikit-engine';
186
+ import {
187
+ outsideClick,
188
+ getBoundingClientRect,
189
+ getScrollInfo,
190
+ } from '@tencentcloud/universal-api';
191
+ import { throttle } from 'lodash';
192
+ import MessageText from './message-elements/message-text.vue';
193
+ import MessageImage from './message-elements/message-image-web.vue';
194
+ import MessageAudio from './message-elements/message-audio-web.vue';
195
+ import MessageRecord from './message-elements/message-record/index.vue';
196
+ import MessageFile from './message-elements/message-file.vue';
197
+ import MessageFace from './message-elements/message-face.vue';
198
+ import MessageCustom from './message-elements/message-custom.vue';
199
+ import MessageBubble from './message-elements/message-bubble-web.vue';
200
+ import MessageLocation from './message-elements/message-location.vue';
201
+ import MessageTimestamp from './message-elements/message-timestamp.vue';
202
+ import MessageVideo from './message-elements/message-video-web.vue';
203
+ import MessageTool from './message-tool/index-web.vue';
204
+ import MessageRevoked from './message-tool/message-revoked.vue';
205
+ import MessagePlugin from '../message-list/message-elements/plugins/plugin-components/message-plugin-web.vue';
206
+ import ScrollButton from './scroll-button/index.vue';
207
+ import { isPluginMessage } from '../message-list/message-elements/plugins/plugin-components/index';
208
+ import Dialog from '../../common/Dialog/index.vue';
209
+ import ImagePreviewer from '../../common/ImagePreviewer/index-web.vue';
210
+ import ProgressMessage from '../../common/ProgressMessage/index.vue';
211
+ import { isPC, isH5 } from '../../../utils/env';
212
+ import chatStorage from '../../../utils/chatStorage';
213
+ import {
214
+ isEnabledMessageReadReceiptGlobal,
215
+ deepCopy,
216
+ } from '../../../utils/utils';
217
+ import { isMessageInvisible } from '../../../utils/index';
218
+ // import {TUICustomerServer} from '../../../server';
219
+ import {isCustomerConversation,getCustomerServiceAccounts,setCustomerServiceAccounts} from '../../../index';
220
+
221
+ interface ScrollConfig {
222
+ scrollToMessage?: IMessageModel;
223
+ scrollToBottom?: boolean;
224
+ scrollToOffset?: {
225
+ top?: number;
226
+ bottom?: number;
227
+ };
228
+ }
229
+
230
+ interface IEmits {
231
+ (key: 'closeInputToolBar'): void;
232
+ (key: 'handleEditor', message: IMessageModel, type: string): void;
233
+ }
234
+
235
+ const emits = defineEmits<IEmits>();
236
+
237
+ let observer: IntersectionObserver | null = null;
238
+ const sentReceiptMessageIDSet = new Set<string>();
239
+ const messageListRef = ref<HTMLElement>();
240
+ const messageToolListRef = ref<Array<{ messageToolDom: HTMLElement }>>();
241
+ // The messageList displayed on the screen, not including messages where isDeleted is true
242
+ const messageList = ref<IMessageModel[]>();
243
+ // All messageList, including messages where isDeleted is true
244
+ const allMessageList = ref<IMessageModel[]>();
245
+ const isCompleted = ref(false);
246
+ const currentConversationID = ref('');
247
+ const currentLastMessage = ref<IMessageModel>();
248
+ const nextReqMessageID = ref();
249
+ const toggleID = ref('');
250
+ const TYPES = ref(TUIChatEngine.TYPES);
251
+ const isLongpressing = ref(false);
252
+ const messageTarget = ref<IMessageModel>();
253
+ const messageElementListRef = ref<HTMLElement[] | null>();
254
+ const targetMessageDom = ref<HTMLElement | null>();
255
+ const blinkMessageIDList = ref<string[]>([]);
256
+ const scrollButtonInstanceRef = ref<InstanceType<typeof ScrollButton>>();
257
+ const beforeHistoryGetScrollHeight = ref<number>(0);
258
+ const isTopMessageDom = ref<boolean>(false);
259
+ const audioPlayedMapping = ref<Record<string, boolean>>({});
260
+
261
+ // image preview
262
+ const showImagePreview = ref(false);
263
+ const currentImagePreview = ref<IMessageModel>();
264
+ const imageMessageList = computed(() =>
265
+ messageList?.value?.filter((item: IMessageModel) => {
266
+ return (
267
+ !item.isRevoked
268
+ && !item.hasRiskContent
269
+ && item.type === TYPES.value.MSG_IMAGE
270
+ );
271
+ }),
272
+ );
273
+
274
+ // resend message dialog
275
+ const reSendDialogShow = ref(false);
276
+ const resendMessageData = ref();
277
+
278
+ onMounted(() => {
279
+ // Retrieve the information about whether the audio has been played from localStorage
280
+ audioPlayedMapping.value
281
+ = chatStorage.getChatStorage('audioPlayedMapping') || {};
282
+
283
+ TUIStore.watch(StoreName.CONV, {
284
+ currentConversationID: onCurrentConversationIDUpdated,
285
+ });
286
+
287
+ TUIStore.watch(StoreName.CHAT, {
288
+ messageList: onMessageListUpdated,
289
+ messageSource: onMessageSourceUpdated,
290
+ isCompleted: isCompletedUpdated,
291
+ });
292
+
293
+
294
+ TUIStore.watch(StoreName.CUSTOM, {
295
+ isShowMessagePopMenu: isShowMessagePopMenuUpdated,
296
+ });
297
+ });
298
+
299
+ onMounted(() => {
300
+ messageListRef.value?.addEventListener('scroll', handelScrollListScroll);
301
+ });
302
+
303
+ onUnmounted(() => {
304
+ TUIStore.unwatch(StoreName.CHAT, {
305
+ messageList: onMessageListUpdated,
306
+ messageSource: onMessageSourceUpdated,
307
+ isCompleted: isCompletedUpdated,
308
+ });
309
+
310
+ TUIStore.unwatch(StoreName.CONV, {
311
+ currentConversationID: onCurrentConversationIDUpdated,
312
+ });
313
+
314
+ TUIStore.unwatch(StoreName.CUSTOM, {
315
+ isShowMessagePopMenu: isShowMessagePopMenuUpdated,
316
+ });
317
+
318
+ messageListRef.value?.removeEventListener('scroll', handelScrollListScroll);
319
+
320
+ if (Object.keys(audioPlayedMapping.value).length > 0) {
321
+ // Synchronize storage about whether the audio has been played when the component is unmounted
322
+ chatStorage.setChatStorage('audioPlayedMapping', audioPlayedMapping.value);
323
+ }
324
+
325
+ sentReceiptMessageIDSet.clear();
326
+ observer?.disconnect();
327
+ observer = null;
328
+ });
329
+
330
+ async function onMessageListUpdated(list: IMessageModel[]) {
331
+ let customeraccounts = getCustomerServiceAccounts();
332
+ if(customeraccounts.length == 0){
333
+ setCustomerServiceAccounts(TUIStore.getData(StoreName.CUSTOM, "customerServiceAccounts"));
334
+ }
335
+ if(!isCustomerConversation(currentConversationID.value)){
336
+ return;
337
+ }
338
+ observer?.disconnect();
339
+ const oldLastMessage = currentLastMessage.value;
340
+ let hasEmojiReaction = false;
341
+ allMessageList.value = list;
342
+ messageList.value = list.filter((message) => {
343
+ if (message.reactionList?.length && !message.isDeleted) {
344
+ hasEmojiReaction = true;
345
+ }
346
+ return !message.isDeleted && !isMessageInvisible(message as any);
347
+ });
348
+ if (!messageList.value?.length) {
349
+ currentLastMessage.value = {};
350
+ return;
351
+ }
352
+ const newLastMessage = messageList.value?.[messageList.value?.length - 1];
353
+ if (messageTarget.value) {
354
+ if (
355
+ messageList.value?.findIndex(
356
+ (message: IMessageModel) => message?.ID === messageTarget.value?.ID,
357
+ ) >= 0
358
+ ) {
359
+ const tempMessage = messageTarget.value;
360
+ messageTarget.value = undefined;
361
+ await scrollToPosition({ scrollToMessage: tempMessage });
362
+ await blinkMessage(tempMessage?.ID);
363
+ }
364
+ } else if (beforeHistoryGetScrollHeight.value) {
365
+ await scrollToPosition({
366
+ scrollToOffset: { bottom: beforeHistoryGetScrollHeight.value },
367
+ });
368
+ beforeHistoryGetScrollHeight.value = 0;
369
+ } else if (
370
+ scrollButtonInstanceRef.value?.isScrollButtonVisible
371
+ && newLastMessage?.flow === 'in'
372
+ ) {
373
+ return;
374
+ } else if (
375
+ newLastMessage?.ID
376
+ && JSON.stringify(oldLastMessage) !== JSON.stringify(newLastMessage)
377
+ ) {
378
+ await scrollToPosition({ scrollToBottom: true });
379
+ } else if (hasEmojiReaction && isCurrentListInBottomPosition()) {
380
+ await scrollToPosition({ scrollToBottom: true });
381
+ }
382
+ currentLastMessage.value = Object.assign({}, newLastMessage);
383
+ if (isEnabledMessageReadReceiptGlobal()) {
384
+ nextTick(() => bindIntersectionObserver());
385
+ }
386
+ }
387
+
388
+ function isCurrentListInBottomPosition() {
389
+ return (
390
+ messageListRef.value
391
+ && typeof messageListRef.value.scrollTop === 'number'
392
+ && typeof messageListRef.value.scrollHeight === 'number'
393
+ && typeof messageListRef.value.clientHeight === 'number'
394
+ && Math.ceil(
395
+ messageListRef.value.scrollTop + messageListRef.value.clientHeight,
396
+ ) >= messageListRef.value.scrollHeight
397
+ );
398
+ }
399
+
400
+ async function scrollToPosition(config: ScrollConfig = {}): Promise<void> {
401
+ return new Promise((resolve, reject) => {
402
+ requestAnimationFrame(() => {
403
+ if (!messageListRef.value) {
404
+ reject();
405
+ }
406
+ const container = messageListRef.value;
407
+ if (config.scrollToBottom && container) {
408
+ container!.scrollTop = container!.scrollHeight;
409
+ } else if (config.scrollToMessage) {
410
+ const targetMessageDom = messageElementListRef.value?.find(
411
+ (dom: HTMLElement) => dom?.id === `tui-${config.scrollToMessage?.ID}`,
412
+ );
413
+ if (targetMessageDom?.scrollIntoView) {
414
+ targetMessageDom.scrollIntoView({
415
+ behavior: 'smooth',
416
+ block: 'nearest',
417
+ });
418
+ }
419
+ } else if (config.scrollToOffset) {
420
+ if (config.scrollToOffset?.top) {
421
+ container!.scrollTop = config.scrollToOffset.top;
422
+ } else if (config.scrollToOffset?.bottom) {
423
+ container!.scrollTop
424
+ = container!.scrollHeight - config.scrollToOffset.bottom;
425
+ }
426
+ }
427
+ resolve();
428
+ });
429
+ });
430
+ }
431
+
432
+ async function onMessageSourceUpdated(message: IMessageModel) {
433
+ // messageSource change has two cases
434
+ // 1. messageSource change -> cache miss -> messageList change,
435
+ // 2. messageSource change -> cache hit -> messageList not change
436
+ // Only the second case needs to add scrollToTarget when listening here
437
+ messageTarget.value = message;
438
+ if (messageTarget.value) {
439
+ if (
440
+ messageList.value?.findIndex(
441
+ (message: IMessageModel) => message?.ID === messageTarget.value?.ID,
442
+ ) >= 0
443
+ ) {
444
+ const tempMessage = messageTarget.value;
445
+ messageTarget.value = undefined;
446
+ await scrollToPosition({ scrollToMessage: tempMessage });
447
+ await blinkMessage(tempMessage?.ID);
448
+ }
449
+ }
450
+ }
451
+
452
+ function isCompletedUpdated(flag: boolean) {
453
+ isCompleted.value = flag;
454
+ }
455
+
456
+ function isShowMessagePopMenuUpdated(isShow: boolean) {
457
+ if (!isShow) {
458
+ toggleID.value = '';
459
+ }
460
+ }
461
+
462
+ const onCurrentConversationIDUpdated = (conversationID: string) => {
463
+ currentConversationID.value = conversationID;
464
+ if (!currentConversationID.value) {
465
+ messageList.value = [];
466
+ }
467
+ if (Object.keys(audioPlayedMapping.value).length > 0) {
468
+ // Synchronize storage about whether the audio has been played when converstaion switched
469
+ chatStorage.setChatStorage('audioPlayedMapping', audioPlayedMapping.value);
470
+ }
471
+ };
472
+
473
+ const getHistoryMessageList = () => {
474
+ TUIChatService.getMessageList().then((res: any) => {
475
+ const { nextReqMessageID: ID } = res.data;
476
+ nextReqMessageID.value = ID;
477
+ });
478
+ // After getting the historical messages, keep the scroll bar in the original position
479
+ beforeHistoryGetScrollHeight.value = messageListRef.value?.scrollHeight;
480
+ };
481
+
482
+ const handleImagePreview = (message: IMessageModel) => {
483
+ if (
484
+ showImagePreview.value
485
+ || currentImagePreview.value
486
+ || isLongpressing.value
487
+ ) {
488
+ return;
489
+ }
490
+ showImagePreview.value = true;
491
+ currentImagePreview.value = message;
492
+ };
493
+
494
+ const onImagePreviewerClose = () => {
495
+ showImagePreview.value = false;
496
+ currentImagePreview.value = null;
497
+ };
498
+
499
+ // toggle message
500
+ const handleToggleMessageItem = (
501
+ e: any,
502
+ message: IMessageModel,
503
+ isLongpress = false,
504
+ ) => {
505
+ if (isLongpress) {
506
+ isLongpressing.value = true;
507
+ }
508
+ toggleID.value = message.ID;
509
+ filterTopMessageDom(e.target);
510
+ };
511
+
512
+ const handleToggleMessageItemForPC = (
513
+ e: MouseEvent,
514
+ message: IMessageModel,
515
+ ) => {
516
+ if (isPC) {
517
+ toggleID.value = message.ID;
518
+ targetMessageDom.value = messageElementListRef.value?.find(
519
+ (dom: HTMLElement) => dom?.id === `tui-${message.ID}`,
520
+ );
521
+ nextTick(() => {
522
+ const ignoreDomRefs
523
+ = messageToolListRef.value && messageToolListRef.value[0]?.messageToolDom;
524
+ outsideClick.listen({
525
+ domRefs: targetMessageDom.value,
526
+ ignoreDomRefs: ignoreDomRefs,
527
+ handler: closeChatPop,
528
+ button: e.button,
529
+ });
530
+ filterTopMessageDom(e.target);
531
+ });
532
+ }
533
+ };
534
+
535
+ function filterTopMessageDom(toggleMessageElement: any) {
536
+ const chatElement = document.getElementById('tui-chat-main');
537
+ const safeTop = 160;
538
+ const messageElementRect = toggleMessageElement.getBoundingClientRect();
539
+ const ChatElementRect = chatElement.getBoundingClientRect();
540
+ isTopMessageDom.value
541
+ = messageElementRect.top - ChatElementRect.top < safeTop ? true : false;
542
+ }
543
+
544
+ // h5 long press
545
+ let timer: number;
546
+ const handleH5LongPress = (e: any, message: IMessageModel, type: string) => {
547
+ if (!isH5) return;
548
+ function longPressHandler() {
549
+ clearTimeout(timer);
550
+ handleToggleMessageItem(e, message);
551
+ }
552
+ function touchStartHandler() {
553
+ timer = setTimeout(longPressHandler, 500);
554
+ }
555
+ function touchEndHandler() {
556
+ clearTimeout(timer);
557
+ }
558
+ switch (type) {
559
+ case 'touchstart':
560
+ touchStartHandler();
561
+ break;
562
+ case 'touchend':
563
+ touchEndHandler();
564
+ setTimeout(() => {
565
+ isLongpressing.value = false;
566
+ }, 200);
567
+ break;
568
+ }
569
+ };
570
+
571
+ // re-edit message
572
+ const handleEdit = (message: IMessageModel) => {
573
+ emits('handleEditor', message, 'reedit');
574
+ };
575
+
576
+ const resendMessage = (message: IMessageModel) => {
577
+ reSendDialogShow.value = true;
578
+ resendMessageData.value = message;
579
+ };
580
+
581
+ const resendMessageConfirm = () => {
582
+ reSendDialogShow.value = !reSendDialogShow.value;
583
+ const messageModel = resendMessageData.value;
584
+ messageModel.resendMessage();
585
+ };
586
+
587
+ function blinkMessage(messageID: string): Promise<void> {
588
+ return new Promise((resolve) => {
589
+ const index = blinkMessageIDList.value.indexOf(messageID);
590
+ if (index < 0) {
591
+ blinkMessageIDList.value.push(messageID);
592
+ const timer = setTimeout(() => {
593
+ blinkMessageIDList.value.splice(
594
+ blinkMessageIDList.value.indexOf(messageID),
595
+ 1,
596
+ );
597
+ clearTimeout(timer);
598
+ resolve();
599
+ }, 3000);
600
+ }
601
+ });
602
+ }
603
+
604
+ async function scrollToLatestMessage() {
605
+ const { scrollHeight } = await getScrollInfo('#messageScrollList');
606
+ const { height } = await getBoundingClientRect('#messageScrollList');
607
+ if (messageListRef.value) {
608
+ messageListRef.value.scrollTop = scrollHeight - height;
609
+ }
610
+ }
611
+
612
+ const handelScrollListScroll = throttle(
613
+ function (e: Event) {
614
+ scrollButtonInstanceRef.value?.judgeScrollOverOneScreen(e);
615
+ },
616
+ 150,
617
+ { leading: true },
618
+ );
619
+
620
+ async function bindIntersectionObserver() {
621
+ if (
622
+ !messageList.value
623
+ || !messageListRef.value
624
+ || messageList.value.length === 0
625
+ ) {
626
+ return;
627
+ }
628
+
629
+ const mappingFromIDToMessage: Record<
630
+ string,
631
+ {
632
+ msgDom: HTMLElement;
633
+ msgModel: IMessageModel | undefined;
634
+ }
635
+ > = {};
636
+
637
+ observer?.disconnect();
638
+ observer = new IntersectionObserver(
639
+ (entries) => {
640
+ entries.forEach((entry) => {
641
+ const { isIntersecting, target } = entry;
642
+ if (isIntersecting) {
643
+ const { msgDom, msgModel } = mappingFromIDToMessage[target.id];
644
+ if (
645
+ msgModel
646
+ && !msgModel.readReceiptInfo?.isPeerRead
647
+ && !sentReceiptMessageIDSet.has(msgModel.ID)
648
+ ) {
649
+ TUIChatService.sendMessageReadReceipt([msgModel]);
650
+ sentReceiptMessageIDSet.add(msgModel.ID);
651
+ observer?.unobserve(msgDom);
652
+ }
653
+ }
654
+ });
655
+ },
656
+ {
657
+ root: messageListRef.value,
658
+ threshold: 0.7,
659
+ },
660
+ );
661
+
662
+ const arrayOfMessageLi
663
+ = messageListRef.value?.querySelectorAll('.message-li');
664
+ if (arrayOfMessageLi) {
665
+ for (let i = 0; i < arrayOfMessageLi?.length; ++i) {
666
+ const messageElement = arrayOfMessageLi[i] as HTMLElement;
667
+ const matchingMessage = messageList.value.find(
668
+ (message: IMessageModel) => {
669
+ return messageElement.id.slice(4) === message.ID;
670
+ },
671
+ );
672
+ if (
673
+ matchingMessage
674
+ && matchingMessage.needReadReceipt
675
+ && matchingMessage.flow === 'in'
676
+ ) {
677
+ mappingFromIDToMessage[messageElement.id] = {
678
+ msgDom: messageElement,
679
+ msgModel: matchingMessage,
680
+ };
681
+ observer?.observe(messageElement);
682
+ }
683
+ }
684
+ }
685
+ }
686
+
687
+ function closeChatPop() {
688
+ toggleID.value = '';
689
+ }
690
+
691
+ function onMessageListBackgroundClick() {
692
+ emits('closeInputToolBar');
693
+ }
694
+
695
+ function setAudioPlayed(messageID: string) {
696
+ audioPlayedMapping.value = {
697
+ ...audioPlayedMapping.value,
698
+ [messageID]: true,
699
+ };
700
+ }
701
+
702
+ defineExpose({
703
+ scrollToLatestMessage,
704
+ });
705
+ </script>
706
+
707
+ <style lang="scss" scoped src="./style/index.scss"></style>
708
+ <style>
709
+ .row-reverse {
710
+ flex-direction: row-reverse;
711
+ }
712
+ </style>