cd-aichat 1.0.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 (356) hide show
  1. package/README.md +188 -0
  2. package/dist/ailogo.png +0 -0
  3. package/dist/cd-aichat.css +1 -0
  4. package/dist/cd-aichat.es.js +32223 -0
  5. package/dist/cd-aichat.umd.js +279 -0
  6. package/dist/index.css +1 -0
  7. package/package.json +61 -0
  8. package/src/ailogo.png +0 -0
  9. package/src/components/AiChat.test.js +538 -0
  10. package/src/components/AiChat.vue +2206 -0
  11. package/src/components/AiChatWidget.test.js +312 -0
  12. package/src/components/AiChatWidget.vue +963 -0
  13. package/src/components/BurnAfterReadDialog.test.js +121 -0
  14. package/src/components/BurnAfterReadDialog.vue +511 -0
  15. package/src/components/BurnAfterReadMessage.test.js +188 -0
  16. package/src/components/BurnAfterReadMessage.vue +193 -0
  17. package/src/components/BurnIndicator.test.js +101 -0
  18. package/src/components/BurnIndicator.vue +164 -0
  19. package/src/components/ChatWindow.vue +0 -0
  20. package/src/components/MentionList.vue +163 -0
  21. package/src/components/ResourceList.vue +194 -0
  22. package/src/components/RichTextEditor.vue +437 -0
  23. package/src/components/ScheduledSendDialog.vue +476 -0
  24. package/src/components/ScreenshotOverlay.vue +78 -0
  25. package/src/components/SendButtonGroup.test.js +174 -0
  26. package/src/components/SendButtonGroup.vue +166 -0
  27. package/src/components/UserDrawer.vue +0 -0
  28. package/src/components/screenshot/ScreenshotsBackground/getBoundsByPoints.ts +41 -0
  29. package/src/components/screenshot/ScreenshotsBackground/index.scss +27 -0
  30. package/src/components/screenshot/ScreenshotsBackground/index.tsx +145 -0
  31. package/src/components/screenshot/ScreenshotsButton/index.scss +29 -0
  32. package/src/components/screenshot/ScreenshotsButton/index.tsx +58 -0
  33. package/src/components/screenshot/ScreenshotsCanvas/getBoundsByPoints.ts +55 -0
  34. package/src/components/screenshot/ScreenshotsCanvas/getPoints.ts +60 -0
  35. package/src/components/screenshot/ScreenshotsCanvas/index.scss +84 -0
  36. package/src/components/screenshot/ScreenshotsCanvas/index.tsx +277 -0
  37. package/src/components/screenshot/ScreenshotsCanvas/isPointInDraw.ts +35 -0
  38. package/src/components/screenshot/ScreenshotsColor/index.scss +45 -0
  39. package/src/components/screenshot/ScreenshotsColor/index.tsx +39 -0
  40. package/src/components/screenshot/ScreenshotsContext.ts +56 -0
  41. package/src/components/screenshot/ScreenshotsMagnifier/index.scss +61 -0
  42. package/src/components/screenshot/ScreenshotsMagnifier/index.tsx +126 -0
  43. package/src/components/screenshot/ScreenshotsOperations/index.scss +25 -0
  44. package/src/components/screenshot/ScreenshotsOperations/index.tsx +118 -0
  45. package/src/components/screenshot/ScreenshotsOption/index.scss +50 -0
  46. package/src/components/screenshot/ScreenshotsOption/index.tsx +150 -0
  47. package/src/components/screenshot/ScreenshotsSize/index.scss +28 -0
  48. package/src/components/screenshot/ScreenshotsSize/index.tsx +41 -0
  49. package/src/components/screenshot/ScreenshotsSizeColor/index.scss +8 -0
  50. package/src/components/screenshot/ScreenshotsSizeColor/index.tsx +25 -0
  51. package/src/components/screenshot/ScreenshotsTextarea/calculateNodeSize.ts +117 -0
  52. package/src/components/screenshot/ScreenshotsTextarea/index.scss +19 -0
  53. package/src/components/screenshot/ScreenshotsTextarea/index.tsx +96 -0
  54. package/src/components/screenshot/composeImage.ts +57 -0
  55. package/src/components/screenshot/exports.ts +4 -0
  56. package/src/components/screenshot/hooks/useBounds.ts +35 -0
  57. package/src/components/screenshot/hooks/useCall.ts +17 -0
  58. package/src/components/screenshot/hooks/useCanvasContextRef.ts +8 -0
  59. package/src/components/screenshot/hooks/useCanvasMousedown.ts +13 -0
  60. package/src/components/screenshot/hooks/useCanvasMousemove.ts +13 -0
  61. package/src/components/screenshot/hooks/useCanvasMouseup.ts +13 -0
  62. package/src/components/screenshot/hooks/useCursor.ts +34 -0
  63. package/src/components/screenshot/hooks/useDispatcher.ts +8 -0
  64. package/src/components/screenshot/hooks/useDrawSelect.ts +16 -0
  65. package/src/components/screenshot/hooks/useEmiter.ts +61 -0
  66. package/src/components/screenshot/hooks/useHistory.ts +160 -0
  67. package/src/components/screenshot/hooks/useLang.ts +8 -0
  68. package/src/components/screenshot/hooks/useOperation.ts +37 -0
  69. package/src/components/screenshot/hooks/useReset.ts +26 -0
  70. package/src/components/screenshot/hooks/useStore.ts +8 -0
  71. package/src/components/screenshot/icons/iconfont.scss +88 -0
  72. package/src/components/screenshot/icons/iconfont.ttf +0 -0
  73. package/src/components/screenshot/icons/iconfont.woff +0 -0
  74. package/src/components/screenshot/icons/iconfont.woff2 +0 -0
  75. package/src/components/screenshot/index.tsx +169 -0
  76. package/src/components/screenshot/operations/Arrow/draw.ts +56 -0
  77. package/src/components/screenshot/operations/Arrow/index.tsx +193 -0
  78. package/src/components/screenshot/operations/Brush/draw.ts +45 -0
  79. package/src/components/screenshot/operations/Brush/index.tsx +169 -0
  80. package/src/components/screenshot/operations/Cancel/index.tsx +18 -0
  81. package/src/components/screenshot/operations/Ellipse/draw.ts +96 -0
  82. package/src/components/screenshot/operations/Ellipse/index.tsx +245 -0
  83. package/src/components/screenshot/operations/Mosaic/index.tsx +223 -0
  84. package/src/components/screenshot/operations/Ok/index.tsx +37 -0
  85. package/src/components/screenshot/operations/Pin/index.tsx +37 -0
  86. package/src/components/screenshot/operations/Rectangle/draw.ts +80 -0
  87. package/src/components/screenshot/operations/Rectangle/index.tsx +245 -0
  88. package/src/components/screenshot/operations/Redo/index.tsx +22 -0
  89. package/src/components/screenshot/operations/Save/index.tsx +37 -0
  90. package/src/components/screenshot/operations/Scan/index.tsx +46 -0
  91. package/src/components/screenshot/operations/Search/index.tsx +39 -0
  92. package/src/components/screenshot/operations/Text/index.tsx +307 -0
  93. package/src/components/screenshot/operations/Undo/index.tsx +22 -0
  94. package/src/components/screenshot/operations/index.ts +34 -0
  95. package/src/components/screenshot/operations/utils.ts +34 -0
  96. package/src/components/screenshot/screenshots.scss +13 -0
  97. package/src/components/screenshot/types.ts +53 -0
  98. package/src/components/screenshot/useGetLoadedImage.ts +29 -0
  99. package/src/components/screenshot/var.scss +107 -0
  100. package/src/components/screenshot/zh_CN.ts +37 -0
  101. package/src/emoji/100.gif +0 -0
  102. package/src/emoji/101.gif +0 -0
  103. package/src/emoji/102.gif +0 -0
  104. package/src/emoji/103.gif +0 -0
  105. package/src/emoji/104.gif +0 -0
  106. package/src/emoji/105.gif +0 -0
  107. package/src/emoji/106.gif +0 -0
  108. package/src/emoji/107.gif +0 -0
  109. package/src/emoji/108.gif +0 -0
  110. package/src/emoji/109.gif +0 -0
  111. package/src/emoji/110.gif +0 -0
  112. package/src/emoji/111.gif +0 -0
  113. package/src/emoji/112.gif +0 -0
  114. package/src/emoji/113.gif +0 -0
  115. package/src/emoji/114.gif +0 -0
  116. package/src/emoji/115.gif +0 -0
  117. package/src/emoji/116.gif +0 -0
  118. package/src/emoji/117.gif +0 -0
  119. package/src/emoji/118.gif +0 -0
  120. package/src/emoji/119.gif +0 -0
  121. package/src/emoji/120.gif +0 -0
  122. package/src/emoji/121.gif +0 -0
  123. package/src/emoji/122.gif +0 -0
  124. package/src/emoji/123.gif +0 -0
  125. package/src/emoji/124.gif +0 -0
  126. package/src/emoji/125.gif +0 -0
  127. package/src/emoji/126.gif +0 -0
  128. package/src/emoji/127.gif +0 -0
  129. package/src/emoji/128.gif +0 -0
  130. package/src/emoji/129.gif +0 -0
  131. package/src/emoji/130.gif +0 -0
  132. package/src/emoji/131.gif +0 -0
  133. package/src/emoji/132.gif +0 -0
  134. package/src/emoji/133.gif +0 -0
  135. package/src/emoji/134.gif +0 -0
  136. package/src/emoji/135.gif +0 -0
  137. package/src/emoji/136.gif +0 -0
  138. package/src/emoji/137.gif +0 -0
  139. package/src/emoji/138.gif +0 -0
  140. package/src/emoji/139.gif +0 -0
  141. package/src/emoji/140.gif +0 -0
  142. package/src/emoji/141.gif +0 -0
  143. package/src/emoji/142.gif +0 -0
  144. package/src/emoji/143.gif +0 -0
  145. package/src/emoji/144.gif +0 -0
  146. package/src/emoji/145.gif +0 -0
  147. package/src/emoji/146.gif +0 -0
  148. package/src/emoji/147.gif +0 -0
  149. package/src/emoji/148.gif +0 -0
  150. package/src/emoji/149.gif +0 -0
  151. package/src/emoji/150.gif +0 -0
  152. package/src/emoji/151.gif +0 -0
  153. package/src/emoji/152.gif +0 -0
  154. package/src/emoji/153.gif +0 -0
  155. package/src/emoji/154.gif +0 -0
  156. package/src/emoji/155.gif +0 -0
  157. package/src/emoji/156.gif +0 -0
  158. package/src/emoji/157.gif +0 -0
  159. package/src/emoji/158.gif +0 -0
  160. package/src/emoji/159.gif +0 -0
  161. package/src/emoji/160.gif +0 -0
  162. package/src/emoji/161.gif +0 -0
  163. package/src/emoji/162.gif +0 -0
  164. package/src/emoji/163.gif +0 -0
  165. package/src/emoji/164.gif +0 -0
  166. package/src/emoji/165.gif +0 -0
  167. package/src/emoji/166.gif +0 -0
  168. package/src/emoji/167.gif +0 -0
  169. package/src/emoji/168.gif +0 -0
  170. package/src/emoji/169.gif +0 -0
  171. package/src/emoji/170.gif +0 -0
  172. package/src/emoji/171.gif +0 -0
  173. package/src/emoji/172.gif +0 -0
  174. package/src/emoji/173.gif +0 -0
  175. package/src/emoji/174.gif +0 -0
  176. package/src/emoji/175.gif +0 -0
  177. package/src/emoji/176.gif +0 -0
  178. package/src/emoji/177.gif +0 -0
  179. package/src/emoji/178.gif +0 -0
  180. package/src/emoji/179.gif +0 -0
  181. package/src/emoji/180.gif +0 -0
  182. package/src/emoji/181.gif +0 -0
  183. package/src/emoji/182.gif +0 -0
  184. package/src/emoji/183.gif +0 -0
  185. package/src/emoji/184.gif +0 -0
  186. package/src/emoji/185.gif +0 -0
  187. package/src/emoji/186.gif +0 -0
  188. package/src/emoji/187.gif +0 -0
  189. package/src/emoji/188.gif +0 -0
  190. package/src/emoji/189.gif +0 -0
  191. package/src/emoji/190.gif +0 -0
  192. package/src/emoji/191.gif +0 -0
  193. package/src/emoji/192.gif +0 -0
  194. package/src/emoji/193.gif +0 -0
  195. package/src/emoji/194.gif +0 -0
  196. package/src/emoji/195.gif +0 -0
  197. package/src/emoji/196.gif +0 -0
  198. package/src/emoji/197.gif +0 -0
  199. package/src/emoji/198.gif +0 -0
  200. package/src/emoji/199.gif +0 -0
  201. package/src/emoji/200.png +0 -0
  202. package/src/emoji/201.png +0 -0
  203. package/src/emoji/202.png +0 -0
  204. package/src/emoji/203.png +0 -0
  205. package/src/emoji/204.png +0 -0
  206. package/src/emoji/205.png +0 -0
  207. package/src/emoji/206.png +0 -0
  208. package/src/emoji/207.png +0 -0
  209. package/src/emoji/208.png +0 -0
  210. package/src/emoji/209.png +0 -0
  211. package/src/emoji/210.png +0 -0
  212. package/src/emoji/211.png +0 -0
  213. package/src/emoji/212.png +0 -0
  214. package/src/emoji/213.png +0 -0
  215. package/src/emoji/214.png +0 -0
  216. package/src/emoji/215.png +0 -0
  217. package/src/emoji/216.png +0 -0
  218. package/src/emoji/217.png +0 -0
  219. package/src/emoji/218.png +0 -0
  220. package/src/emoji/219.png +0 -0
  221. package/src/emoji2/101--Streamline-The-Team.png +0 -0
  222. package/src/emoji2/128--Streamline-The-Team.png +0 -0
  223. package/src/emoji2/134--Streamline-The-Team.png +0 -0
  224. package/src/emoji2/173--Streamline-The-Team.png +0 -0
  225. package/src/emoji2/Airplane--Streamline-Emoji.svg +24 -0
  226. package/src/emoji2/Alien--Streamline-Emoji.svg +21 -0
  227. package/src/emoji2/Amazed-Face--Streamline-Emoji.svg +16 -0
  228. package/src/emoji2/Amusing-Face--Streamline-Emoji.svg +20 -0
  229. package/src/emoji2/Anguished-Face--Streamline-Emoji.svg +19 -0
  230. package/src/emoji2/Anxious-Face--Streamline-Emoji.svg +17 -0
  231. package/src/emoji2/Astonished-Face--Streamline-Emoji.svg +20 -0
  232. package/src/emoji2/Backhand-Index-Pointing-Down-1--Streamline-Emoji.svg +12 -0
  233. package/src/emoji2/Backhand-Index-Pointing-Left-1--Streamline-Emoji.svg +13 -0
  234. package/src/emoji2/Backhand-Index-Pointing-Right-1--Streamline-Emoji.svg +13 -0
  235. package/src/emoji2/Backhand-Index-Pointing-Up-1--Streamline-Emoji.svg +14 -0
  236. package/src/emoji2/Bar-Chart--Streamline-Emoji.svg +22 -0
  237. package/src/emoji2/Beaming-Face-With-Smiling-Eyes--Streamline-Emoji.svg +20 -0
  238. package/src/emoji2/Boy-1--Streamline-Emoji.svg +17 -0
  239. package/src/emoji2/Boy-2--Streamline-Emoji.svg +17 -0
  240. package/src/emoji2/Boy-3--Streamline-Emoji.svg +17 -0
  241. package/src/emoji2/Broken-Heart--Streamline-Emoji.svg +12 -0
  242. package/src/emoji2/Clapping-Hands-1--Streamline-Emoji.svg +23 -0
  243. package/src/emoji2/Clinking-Glasses-2--Streamline-Emoji.svg +43 -0
  244. package/src/emoji2/Confounded-Face--Streamline-Emoji.svg +17 -0
  245. package/src/emoji2/Confused-Face--Streamline-Emoji.svg +15 -0
  246. package/src/emoji2/Construction-Worker--Streamline-Emoji.svg +21 -0
  247. package/src/emoji2/Couple-With-Heart-Woman-Man-1--Streamline-Emoji.svg +40 -0
  248. package/src/emoji2/Couple-With-Heart-Woman-Man-2--Streamline-Emoji.svg +40 -0
  249. package/src/emoji2/Cowboy-Hat-Face--Streamline-Emoji.svg +22 -0
  250. package/src/emoji2/Crazy-Face--Streamline-Emoji.svg +25 -0
  251. package/src/emoji2/Crossed-Fingers-1--Streamline-Emoji.svg +25 -0
  252. package/src/emoji2/Crown--Streamline-Emoji.svg +35 -0
  253. package/src/emoji2/Crying-Face--Streamline-Emoji.svg +26 -0
  254. package/src/emoji2/Delivery-Truck--Streamline-Emoji.svg +31 -0
  255. package/src/emoji2/Determined-Face--Streamline-Emoji.svg +25 -0
  256. package/src/emoji2/Disappointed-Face--Streamline-Emoji.svg +15 -0
  257. package/src/emoji2/Dizzy-Face--Streamline-Emoji.svg +20 -0
  258. package/src/emoji2/Downcast-Face-With-Sweat--Streamline-Emoji.svg +18 -0
  259. package/src/emoji2/Drooling-Face-1--Streamline-Emoji.svg +19 -0
  260. package/src/emoji2/Drooling-Face-2--Streamline-Emoji.svg +18 -0
  261. package/src/emoji2/Ear--Streamline-Emoji.svg +14 -0
  262. package/src/emoji2/Exclamation-Mark--Streamline-Emoji.svg +12 -0
  263. package/src/emoji2/Exploding-Head--Streamline-Emoji.svg +24 -0
  264. package/src/emoji2/Expressionless-Face--Streamline-Emoji.svg +15 -0
  265. package/src/emoji2/Face-Blowing-A-Kiss--Streamline-Emoji.svg +18 -0
  266. package/src/emoji2/Face-Savoring-Food--Streamline-Emoji.svg +18 -0
  267. package/src/emoji2/Face-Screaming-In-Fear--Streamline-Emoji.svg +18 -0
  268. package/src/emoji2/Face-Vomiting--Streamline-Emoji.svg +28 -0
  269. package/src/emoji2/Face-With-Head-Bandage--Streamline-Emoji.svg +25 -0
  270. package/src/emoji2/Face-With-Medical-Mask--Streamline-Emoji.svg +23 -0
  271. package/src/emoji2/Face-With-Monocle--Streamline-Emoji.svg +26 -0
  272. package/src/emoji2/Face-With-Raised-Eyebrow--Streamline-Emoji.svg +17 -0
  273. package/src/emoji2/Face-With-Rolling-Eyes--Streamline-Emoji.svg +17 -0
  274. package/src/emoji2/Face-With-Steam-From-Nose--Streamline-Emoji.svg +19 -0
  275. package/src/emoji2/Face-With-Symbols-On-Mouth--Streamline-Emoji.svg +22 -0
  276. package/src/emoji2/Face-With-Tears-Of-Joy--Streamline-Emoji.svg +34 -0
  277. package/src/emoji2/Face-With-Thermometer--Streamline-Emoji.svg +31 -0
  278. package/src/emoji2/Face-With-Tongue--Streamline-Emoji.svg +18 -0
  279. package/src/emoji2/Face-Without-Mouth--Streamline-Emoji.svg +14 -0
  280. package/src/emoji2/Fearful-Face--Streamline-Emoji.svg +18 -0
  281. package/src/emoji2/Flexed-Biceps-1--Streamline-Emoji.svg +13 -0
  282. package/src/emoji2/Flushed-Face--Streamline-Emoji.svg +19 -0
  283. package/src/emoji2/Folded-Hands-1--Streamline-Emoji.svg +29 -0
  284. package/src/emoji2/Frowning-Face--Streamline-Emoji.svg +15 -0
  285. package/src/emoji2/Fuel-Pump--Streamline-Emoji.svg +30 -0
  286. package/src/emoji2/Girl-1--Streamline-Emoji.svg +23 -0
  287. package/src/emoji2/Glasses-1--Streamline-Emoji.svg +27 -0
  288. package/src/emoji2/Grimacing-Face--Streamline-Emoji.svg +19 -0
  289. package/src/emoji2/Grinning-Cat-Face--Streamline-Emoji.svg +32 -0
  290. package/src/emoji2/Grinning-Face--Streamline-Emoji.svg +16 -0
  291. package/src/emoji2/Grinning-Face-With-Sweat--Streamline-Emoji.svg +19 -0
  292. package/src/emoji2/Grinning-Squinting-Face--Streamline-Emoji.svg +16 -0
  293. package/src/emoji2/Hand-With-Fingers-Splayed-1--Streamline-Emoji.svg +14 -0
  294. package/src/emoji2/Heart-Suit--Streamline-Emoji.svg +9 -0
  295. package/src/emoji2/Hushed-Face-1--Streamline-Emoji.svg +17 -0
  296. package/src/emoji2/Hushed-Face-2--Streamline-Emoji.svg +15 -0
  297. package/src/emoji2/Index-Pointing-Up-1--Streamline-Emoji.svg +17 -0
  298. package/src/emoji2/Kissing-Face-With-Closed-Eyes--Streamline-Emoji.svg +18 -0
  299. package/src/emoji2/Loudly-Crying-Face--Streamline-Emoji.svg +16 -0
  300. package/src/emoji2/Lying-Face--Streamline-Emoji.svg +15 -0
  301. package/src/emoji2/Man-1--Streamline-Emoji.svg +17 -0
  302. package/src/emoji2/Man-Facepalming-1--Streamline-Emoji.svg +21 -0
  303. package/src/emoji2/Man-Gesturing-No-1--Streamline-Emoji.svg +34 -0
  304. package/src/emoji2/Man-Gesturing-Ok-1--Streamline-Emoji.svg +25 -0
  305. package/src/emoji2/Man-Health-Worker-1--Streamline-Emoji.svg +41 -0
  306. package/src/emoji2/Man-Raising-Hand-1--Streamline-Emoji.svg +26 -0
  307. package/src/emoji2/Man-Shrugging-1--Streamline-Emoji.svg +31 -0
  308. package/src/emoji2/Money-Mouth-Face-2--Streamline-Emoji.svg +30 -0
  309. package/src/emoji2/Mouth--Streamline-Emoji.svg +12 -0
  310. package/src/emoji2/Nauseated-Face-2--Streamline-Emoji.svg +19 -0
  311. package/src/emoji2/Neutral-Face--Streamline-Emoji.svg +15 -0
  312. package/src/emoji2/Ok-Hand-1--Streamline-Emoji.svg +14 -0
  313. package/src/emoji2/Old-Man-1--Streamline-Emoji.svg +27 -0
  314. package/src/emoji2/Old-Woman-1--Streamline-Emoji.svg +23 -0
  315. package/src/emoji2/Oncoming-Fist-1--Streamline-Emoji.svg +15 -0
  316. package/src/emoji2/Person-Wearing-Turban-2--Streamline-Emoji.svg +20 -0
  317. package/src/emoji2/Pile-Of-Poo--Streamline-Emoji.svg +15 -0
  318. package/src/emoji2/Police-Car-Light--Streamline-Emoji.svg +26 -0
  319. package/src/emoji2/Rocket--Streamline-Emoji.svg +32 -0
  320. package/src/emoji2/Sailboat--Streamline-Emoji.svg +18 -0
  321. package/src/emoji2/Shaved-Ice--Streamline-Emoji.svg +21 -0
  322. package/src/emoji2/Shortcake-2--Streamline-Emoji.svg +18 -0
  323. package/src/emoji2/Shushing-Face--Streamline-Emoji.svg +21 -0
  324. package/src/emoji2/Sign-Of-The-Horns-1--Streamline-Emoji.svg +19 -0
  325. package/src/emoji2/Sleeping-Face--Streamline-Emoji.svg +21 -0
  326. package/src/emoji2/Slightly-Smiling-Face--Streamline-Emoji.svg +15 -0
  327. package/src/emoji2/Smiling-Face-With-Halo--Streamline-Emoji.svg +20 -0
  328. package/src/emoji2/Smiling-Face-With-Heart-Eyes--Streamline-Emoji.svg +21 -0
  329. package/src/emoji2/Smirking-Face--Streamline-Emoji.svg +17 -0
  330. package/src/emoji2/Sun-With-Face--Streamline-Emoji.svg +24 -0
  331. package/src/emoji2/Thumbs-Down-1--Streamline-Emoji.svg +20 -0
  332. package/src/emoji2/Thumbs-Up-1--Streamline-Emoji.svg +19 -0
  333. package/src/emoji2/Winking-Face--Streamline-Emoji.svg +18 -0
  334. package/src/emoji2/Woman-Gesturing-No-1--Streamline-Emoji.svg +34 -0
  335. package/src/emoji2/Woman-Gesturing-Ok-2--Streamline-Emoji.svg +25 -0
  336. package/src/emoji2/Woman-Raising-Hand-1--Streamline-Emoji.svg +26 -0
  337. package/src/emoji2/Womans-Sandal--Streamline-Emoji.svg +13 -0
  338. package/src/emoji2/Worried-Face--Streamline-Emoji.svg +17 -0
  339. package/src/emoji2/Writing-Hand-1--Streamline-Emoji.svg +17 -0
  340. package/src/emoji2/Zipper-Mouth-Face--Streamline-Emoji.svg +21 -0
  341. package/src/index.js +19 -0
  342. package/src/services/burn-after-read-service.js +313 -0
  343. package/src/services/burn-after-read-service.test.js +325 -0
  344. package/src/services/dify-api.js +338 -0
  345. package/src/services/dify-api.test.js +376 -0
  346. package/src/services/scheduled-send-service.js +311 -0
  347. package/src/services/scheduled-send-service.test.js +317 -0
  348. package/src/styles/index.css +2368 -0
  349. package/src/utils/emoji.js +125 -0
  350. package/src/utils/emojiData.js +267 -0
  351. package/src/utils/eventEmitter.js +114 -0
  352. package/src/utils/state.js +224 -0
  353. package/src/utils/state.test.js +198 -0
  354. package/src/utils/storage.js +122 -0
  355. package/src/utils/storage.test.js +162 -0
  356. package/src/utils/validation.js +249 -0
@@ -0,0 +1,188 @@
1
+ import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest';
2
+ import { mount, flushPromises } from '@vue/test-utils';
3
+ import BurnAfterReadMessage from './BurnAfterReadMessage.vue';
4
+ import { resetBurnAfterReadService } from '../services/burn-after-read-service';
5
+
6
+ describe('BurnAfterReadMessage.vue', () => {
7
+ beforeEach(() => {
8
+ vi.useFakeTimers();
9
+ resetBurnAfterReadService();
10
+ });
11
+
12
+ afterEach(() => {
13
+ vi.useRealTimers();
14
+ });
15
+
16
+ it('应该渲染消息内容', () => {
17
+ const wrapper = mount(BurnAfterReadMessage, {
18
+ props: {
19
+ messageId: 'msg1',
20
+ burnDelay: 10
21
+ },
22
+ slots: {
23
+ default: 'Test message content'
24
+ }
25
+ });
26
+ expect(wrapper.text()).toContain('Test message content');
27
+ });
28
+
29
+ it('应该显示阅后即焚指示器', () => {
30
+ const wrapper = mount(BurnAfterReadMessage, {
31
+ props: {
32
+ messageId: 'msg1',
33
+ burnDelay: 10
34
+ },
35
+ slots: {
36
+ default: 'Test message'
37
+ }
38
+ });
39
+ expect(wrapper.findComponent({ name: 'BurnIndicator' }).exists()).toBe(true);
40
+ });
41
+
42
+ it('应该在销毁后显示占位符', async () => {
43
+ const wrapper = mount(BurnAfterReadMessage, {
44
+ props: {
45
+ messageId: 'msg1',
46
+ burnDelay: 1
47
+ },
48
+ slots: {
49
+ default: 'Test message'
50
+ }
51
+ });
52
+
53
+ // 标记为已读
54
+ wrapper.vm.markAsRead();
55
+ await flushPromises();
56
+
57
+ // 快进时间
58
+ vi.advanceTimersByTime(1000);
59
+ await flushPromises();
60
+
61
+ expect(wrapper.text()).toContain('消息已销毁');
62
+ expect(wrapper.text()).not.toContain('Test message');
63
+ });
64
+
65
+ it('应该发出 read 事件', async () => {
66
+ const wrapper = mount(BurnAfterReadMessage, {
67
+ props: {
68
+ messageId: 'msg1',
69
+ burnDelay: 10
70
+ },
71
+ slots: {
72
+ default: 'Test message'
73
+ }
74
+ });
75
+
76
+ wrapper.vm.markAsRead();
77
+ await flushPromises();
78
+
79
+ expect(wrapper.emitted('read')).toBeTruthy();
80
+ expect(wrapper.emitted('read')[0][0].messageId).toBe('msg1');
81
+ });
82
+
83
+ it('应该发出 burned 事件', async () => {
84
+ const wrapper = mount(BurnAfterReadMessage, {
85
+ props: {
86
+ messageId: 'msg1',
87
+ burnDelay: 1
88
+ },
89
+ slots: {
90
+ default: 'Test message'
91
+ }
92
+ });
93
+
94
+ wrapper.vm.markAsRead();
95
+ await flushPromises();
96
+
97
+ vi.advanceTimersByTime(1000);
98
+ await flushPromises();
99
+
100
+ expect(wrapper.emitted('burned')).toBeTruthy();
101
+ expect(wrapper.emitted('burned')[0][0].messageId).toBe('msg1');
102
+ });
103
+
104
+ it('应该在卸载时清理资源', async () => {
105
+ const wrapper = mount(BurnAfterReadMessage, {
106
+ props: {
107
+ messageId: 'msg1',
108
+ burnDelay: 10
109
+ },
110
+ slots: {
111
+ default: 'Test message'
112
+ }
113
+ });
114
+
115
+ wrapper.vm.markAsRead();
116
+ await flushPromises();
117
+
118
+ wrapper.unmount();
119
+
120
+ // 快进时间,不应该触发销毁事件
121
+ vi.advanceTimersByTime(10000);
122
+ await flushPromises();
123
+
124
+ // 验证没有内存泄漏
125
+ expect(wrapper.emitted('burned')).toBeFalsy();
126
+ });
127
+
128
+ it('应该显示倒计时', async () => {
129
+ const wrapper = mount(BurnAfterReadMessage, {
130
+ props: {
131
+ messageId: 'msg1',
132
+ burnDelay: 10,
133
+ showCountdown: true
134
+ },
135
+ slots: {
136
+ default: 'Test message'
137
+ }
138
+ });
139
+
140
+ wrapper.vm.markAsRead();
141
+ await flushPromises();
142
+
143
+ // 快进3秒
144
+ vi.advanceTimersByTime(3000);
145
+ await flushPromises();
146
+
147
+ const indicator = wrapper.findComponent({ name: 'BurnIndicator' });
148
+ expect(indicator.props('remainingTime')).toBeLessThanOrEqual(7);
149
+ });
150
+
151
+ it('应该在 showCountdown 为 false 时隐藏倒计时', () => {
152
+ const wrapper = mount(BurnAfterReadMessage, {
153
+ props: {
154
+ messageId: 'msg1',
155
+ burnDelay: 10,
156
+ showCountdown: false
157
+ },
158
+ slots: {
159
+ default: 'Test message'
160
+ }
161
+ });
162
+
163
+ const indicator = wrapper.findComponent({ name: 'BurnIndicator' });
164
+ expect(indicator.props('showCountdown')).toBe(false);
165
+ });
166
+
167
+ it('应该添加 burned 类当消息被销毁', async () => {
168
+ const wrapper = mount(BurnAfterReadMessage, {
169
+ props: {
170
+ messageId: 'msg1',
171
+ burnDelay: 1
172
+ },
173
+ slots: {
174
+ default: 'Test message'
175
+ }
176
+ });
177
+
178
+ expect(wrapper.find('.burn-after-read-message--burned').exists()).toBe(false);
179
+
180
+ wrapper.vm.markAsRead();
181
+ await flushPromises();
182
+
183
+ vi.advanceTimersByTime(1000);
184
+ await flushPromises();
185
+
186
+ expect(wrapper.find('.burn-after-read-message--burned').exists()).toBe(true);
187
+ });
188
+ });
@@ -0,0 +1,193 @@
1
+ <template>
2
+ <div class="burn-after-read-message" :class="{ 'burn-after-read-message--burned': isBurned }">
3
+ <div class="message-wrapper">
4
+ <div v-if="!isBurned" class="message-content">
5
+ <slot />
6
+ </div>
7
+ <div v-else class="burned-placeholder">
8
+ <i class="ri-fire-line"></i>
9
+ <span>消息已销毁</span>
10
+ </div>
11
+ </div>
12
+ <div class="burn-indicator-wrapper">
13
+ <BurnIndicator
14
+ :status="burnStatus"
15
+ :remaining-time="remainingTime"
16
+ :burn-delay="burnDelay"
17
+ :show-countdown="showCountdown"
18
+ />
19
+ </div>
20
+ </div>
21
+ </template>
22
+
23
+ <script>
24
+ import { ref, computed, onMounted, onUnmounted } from 'vue';
25
+ import BurnIndicator from './BurnIndicator.vue';
26
+ import { getBurnAfterReadService } from '../services/burn-after-read-service';
27
+
28
+ export default {
29
+ name: 'BurnAfterReadMessage',
30
+ components: {
31
+ BurnIndicator
32
+ },
33
+ props: {
34
+ messageId: {
35
+ type: String,
36
+ required: true
37
+ },
38
+ burnDelay: {
39
+ type: Number,
40
+ required: true
41
+ },
42
+ showCountdown: {
43
+ type: Boolean,
44
+ default: true
45
+ }
46
+ },
47
+ emits: ['burned', 'read'],
48
+ setup(props, { emit }) {
49
+ const service = getBurnAfterReadService();
50
+ const burnStatus = ref('pending');
51
+ const remainingTime = ref(0);
52
+ const isBurned = ref(false);
53
+ let updateInterval = null;
54
+ let unsubscribeRead = null;
55
+ let unsubscribeBurned = null;
56
+
57
+ const markAsRead = () => {
58
+ try {
59
+ service.markMessageAsRead(props.messageId);
60
+ burnStatus.value = 'read';
61
+ emit('read', { messageId: props.messageId });
62
+ startCountdownUpdate();
63
+ } catch (error) {
64
+ console.error('Failed to mark message as read:', error);
65
+ }
66
+ };
67
+
68
+ const startCountdownUpdate = () => {
69
+ if (updateInterval) {
70
+ clearInterval(updateInterval);
71
+ }
72
+ updateInterval = setInterval(() => {
73
+ const remaining = service.getRemainingTime(props.messageId);
74
+ if (remaining !== null) {
75
+ remainingTime.value = remaining;
76
+ }
77
+ }, 100);
78
+ };
79
+
80
+ const handleBurned = () => {
81
+ isBurned.value = true;
82
+ burnStatus.value = 'burned';
83
+ if (updateInterval) {
84
+ clearInterval(updateInterval);
85
+ }
86
+ emit('burned', { messageId: props.messageId });
87
+ };
88
+
89
+ const cleanup = () => {
90
+ if (updateInterval) {
91
+ clearInterval(updateInterval);
92
+ }
93
+ if (unsubscribeRead) {
94
+ unsubscribeRead();
95
+ }
96
+ if (unsubscribeBurned) {
97
+ unsubscribeBurned();
98
+ }
99
+ };
100
+
101
+ onMounted(() => {
102
+ // 标记消息为阅后即焚
103
+ try {
104
+ service.markMessageForBurn({
105
+ id: props.messageId,
106
+ burnDelay: props.burnDelay
107
+ });
108
+ } catch (error) {
109
+ console.error('Failed to mark message for burn:', error);
110
+ }
111
+
112
+ // 监听消息被读事件
113
+ unsubscribeRead = service.on('message-read', (data) => {
114
+ if (data.id === props.messageId) {
115
+ burnStatus.value = 'read';
116
+ startCountdownUpdate();
117
+ }
118
+ });
119
+
120
+ // 监听消息销毁事件
121
+ unsubscribeBurned = service.on('message-burned', (data) => {
122
+ if (data.id === props.messageId) {
123
+ handleBurned();
124
+ }
125
+ });
126
+
127
+ // 检查消息是否已经被标记为已读
128
+ const status = service.getBurnStatus(props.messageId);
129
+ if (status && status.status === 'read') {
130
+ burnStatus.value = 'read';
131
+ startCountdownUpdate();
132
+ }
133
+ });
134
+
135
+ onUnmounted(() => {
136
+ cleanup();
137
+ });
138
+
139
+ return {
140
+ burnStatus,
141
+ remainingTime,
142
+ isBurned,
143
+ markAsRead
144
+ };
145
+ }
146
+ };
147
+ </script>
148
+
149
+ <style scoped>
150
+ .burn-after-read-message {
151
+ display: flex;
152
+ align-items: flex-start;
153
+ gap: 8px;
154
+ padding: 8px;
155
+ border-radius: 6px;
156
+ background-color: #fff9f5;
157
+ border: 1px solid #ffe0cc;
158
+ transition: all 0.3s ease;
159
+ }
160
+
161
+ .burn-after-read-message--burned {
162
+ background-color: #f5f5f5;
163
+ border-color: #e0e0e0;
164
+ opacity: 0.7;
165
+ }
166
+
167
+ .message-wrapper {
168
+ flex: 1;
169
+ min-width: 0;
170
+ }
171
+
172
+ .message-content {
173
+ word-break: break-word;
174
+ line-height: 1.5;
175
+ }
176
+
177
+ .burned-placeholder {
178
+ display: flex;
179
+ align-items: center;
180
+ gap: 6px;
181
+ color: #999;
182
+ font-size: 13px;
183
+ font-style: italic;
184
+ }
185
+
186
+ .burned-placeholder i {
187
+ font-size: 14px;
188
+ }
189
+
190
+ .burn-indicator-wrapper {
191
+ flex-shrink: 0;
192
+ }
193
+ </style>
@@ -0,0 +1,101 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { mount } from '@vue/test-utils';
3
+ import BurnIndicator from './BurnIndicator.vue';
4
+
5
+ describe('BurnIndicator.vue', () => {
6
+ it('应该渲染待销毁状态', () => {
7
+ const wrapper = mount(BurnIndicator, {
8
+ props: {
9
+ status: 'pending',
10
+ burnDelay: 10
11
+ }
12
+ });
13
+ expect(wrapper.find('.burn-indicator--pending').exists()).toBe(true);
14
+ expect(wrapper.text()).toContain('阅后即焚');
15
+ });
16
+
17
+ it('应该渲染已读状态并显示倒计时', () => {
18
+ const wrapper = mount(BurnIndicator, {
19
+ props: {
20
+ status: 'read',
21
+ remainingTime: 5,
22
+ showCountdown: true
23
+ }
24
+ });
25
+ expect(wrapper.find('.burn-indicator--read').exists()).toBe(true);
26
+ expect(wrapper.text()).toContain('5秒');
27
+ });
28
+
29
+ it('应该渲染已销毁状态', () => {
30
+ const wrapper = mount(BurnIndicator, {
31
+ props: {
32
+ status: 'burned'
33
+ }
34
+ });
35
+ expect(wrapper.find('.burn-indicator--burned').exists()).toBe(true);
36
+ expect(wrapper.text()).toContain('已销毁');
37
+ });
38
+
39
+ it('应该在倒计时时添加动画类', () => {
40
+ const wrapper = mount(BurnIndicator, {
41
+ props: {
42
+ status: 'read',
43
+ remainingTime: 5
44
+ }
45
+ });
46
+ expect(wrapper.find('.burn-indicator--burning').exists()).toBe(true);
47
+ });
48
+
49
+ it('应该格式化时间为分钟', () => {
50
+ const wrapper = mount(BurnIndicator, {
51
+ props: {
52
+ status: 'read',
53
+ remainingTime: 65,
54
+ showCountdown: true
55
+ }
56
+ });
57
+ expect(wrapper.text()).toContain('1分5秒');
58
+ });
59
+
60
+ it('应该格式化时间为小时', () => {
61
+ const wrapper = mount(BurnIndicator, {
62
+ props: {
63
+ status: 'read',
64
+ remainingTime: 3665,
65
+ showCountdown: true
66
+ }
67
+ });
68
+ expect(wrapper.text()).toContain('1小时1分钟');
69
+ });
70
+
71
+ it('应该在不显示倒计时时隐藏时间', () => {
72
+ const wrapper = mount(BurnIndicator, {
73
+ props: {
74
+ status: 'read',
75
+ remainingTime: 5,
76
+ showCountdown: false
77
+ }
78
+ });
79
+ expect(wrapper.text()).not.toContain('5秒');
80
+ });
81
+
82
+ it('应该生成正确的工具提示文本', () => {
83
+ const wrapper = mount(BurnIndicator, {
84
+ props: {
85
+ status: 'pending',
86
+ burnDelay: 10
87
+ }
88
+ });
89
+ expect(wrapper.attributes('title')).toContain('阅后即焚消息');
90
+ });
91
+
92
+ it('应该在已读状态生成正确的工具提示', () => {
93
+ const wrapper = mount(BurnIndicator, {
94
+ props: {
95
+ status: 'read',
96
+ remainingTime: 5
97
+ }
98
+ });
99
+ expect(wrapper.attributes('title')).toContain('消息将在');
100
+ });
101
+ });
@@ -0,0 +1,164 @@
1
+ <template>
2
+ <div
3
+ class="burn-indicator"
4
+ :class="{
5
+ 'burn-indicator--pending': status === 'pending',
6
+ 'burn-indicator--read': status === 'read',
7
+ 'burn-indicator--burning': status === 'read' && remainingTime > 0,
8
+ 'burn-indicator--burned': status === 'burned'
9
+ }"
10
+ :title="tooltipText"
11
+ >
12
+ <i
13
+ class="burn-icon ri-fire-line"
14
+ :class="{ 'burn-icon--animated': status === 'read' && remainingTime > 0 }"
15
+ ></i>
16
+ <span v-if="showCountdown && remainingTime > 0" class="burn-countdown">
17
+ {{ formatTime(remainingTime) }}
18
+ </span>
19
+ <span v-else-if="status === 'burned'" class="burn-status">
20
+ 已销毁
21
+ </span>
22
+ <span v-else-if="status === 'pending'" class="burn-status">
23
+ 阅后即焚
24
+ </span>
25
+ </div>
26
+ </template>
27
+
28
+ <script>
29
+ import { computed } from 'vue';
30
+
31
+ export default {
32
+ name: 'BurnIndicator',
33
+ props: {
34
+ status: {
35
+ type: String,
36
+ required: true,
37
+ validator: (value) => ['pending', 'read', 'burned'].includes(value)
38
+ },
39
+ remainingTime: {
40
+ type: Number,
41
+ default: 0
42
+ },
43
+ showCountdown: {
44
+ type: Boolean,
45
+ default: true
46
+ },
47
+ burnDelay: {
48
+ type: Number,
49
+ default: 0
50
+ }
51
+ },
52
+ setup(props) {
53
+ const tooltipText = computed(() => {
54
+ switch (props.status) {
55
+ case 'pending':
56
+ return `阅后即焚消息,将在阅读后 ${props.burnDelay} 秒销毁`;
57
+ case 'read':
58
+ if (props.remainingTime > 0) {
59
+ return `消息将在 ${formatTime(props.remainingTime)} 后销毁`;
60
+ }
61
+ return '消息即将销毁';
62
+ case 'burned':
63
+ return '消息已销毁';
64
+ default:
65
+ return '阅后即焚消息';
66
+ }
67
+ });
68
+
69
+ const formatTime = (seconds) => {
70
+ if (seconds < 60) {
71
+ return `${seconds}秒`;
72
+ } else if (seconds < 3600) {
73
+ const minutes = Math.floor(seconds / 60);
74
+ const remainingSeconds = seconds % 60;
75
+ return remainingSeconds > 0
76
+ ? `${minutes}分${remainingSeconds}秒`
77
+ : `${minutes}分钟`;
78
+ } else {
79
+ const hours = Math.floor(seconds / 3600);
80
+ const remainingMinutes = Math.floor((seconds % 3600) / 60);
81
+ return remainingMinutes > 0
82
+ ? `${hours}小时${remainingMinutes}分钟`
83
+ : `${hours}小时`;
84
+ }
85
+ };
86
+
87
+ return {
88
+ tooltipText,
89
+ formatTime
90
+ };
91
+ }
92
+ };
93
+ </script>
94
+
95
+ <style scoped>
96
+ .burn-indicator {
97
+ display: inline-flex;
98
+ align-items: center;
99
+ gap: 4px;
100
+ padding: 2px 6px;
101
+ border-radius: 12px;
102
+ font-size: 11px;
103
+ font-weight: 500;
104
+ transition: all 0.3s ease;
105
+ cursor: help;
106
+ }
107
+
108
+ .burn-indicator--pending {
109
+ background: linear-gradient(135deg, #ff9800 0%, #ff5722 100%);
110
+ color: white;
111
+ }
112
+
113
+ .burn-indicator--read {
114
+ background: linear-gradient(135deg, #ff6b6b 0%, #ff8e53 100%);
115
+ color: white;
116
+ }
117
+
118
+ .burn-indicator--burning {
119
+ animation: pulse 1.5s ease-in-out infinite;
120
+ }
121
+
122
+ .burn-indicator--burned {
123
+ background: linear-gradient(135deg, #9e9e9e 0%, #757575 100%);
124
+ color: white;
125
+ opacity: 0.7;
126
+ }
127
+
128
+ .burn-icon {
129
+ font-size: 12px;
130
+ display: inline-flex;
131
+ align-items: center;
132
+ }
133
+
134
+ .burn-icon--animated {
135
+ animation: burn-flame 1s ease-in-out infinite;
136
+ }
137
+
138
+ @keyframes burn-flame {
139
+ 0%, 100% {
140
+ transform: scale(1) rotate(-2deg);
141
+ }
142
+ 50% {
143
+ transform: scale(1.15) rotate(2deg);
144
+ }
145
+ }
146
+
147
+ @keyframes pulse {
148
+ 0%, 100% {
149
+ opacity: 1;
150
+ }
151
+ 50% {
152
+ opacity: 0.7;
153
+ }
154
+ }
155
+
156
+ .burn-countdown {
157
+ font-variant-numeric: tabular-nums;
158
+ min-width: 30px;
159
+ }
160
+
161
+ .burn-status {
162
+ font-size: 10px;
163
+ }
164
+ </style>
File without changes