@gientech/modual 1.4.2 → 1.4.3

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 (317) hide show
  1. package/.editorconfig +38 -0
  2. package/.prettierignore +16 -0
  3. package/.prettierrc +17 -0
  4. package/README.md +129 -129
  5. package/USAGE.md +191 -0
  6. package/bash.exe.stackdump +40 -0
  7. package/components.json +21 -0
  8. package/dist/README.md +129 -0
  9. package/{assets → dist/assets}/database.svg +11 -11
  10. package/{assets → dist/assets}/database_add.svg +53 -53
  11. package/{assets → dist/assets}/database_connect.svg +66 -66
  12. package/{assets → dist/assets}/database_upload.svg +29 -29
  13. package/{assets → dist/assets}/defaultWeLogo.svg +14 -14
  14. package/{assets → dist/assets}/mysql.svg +14 -14
  15. package/dist/database.js +20 -0
  16. package/dist/package.json +63 -0
  17. package/{worker → dist/worker}/pdf.worker.min.js +21 -21
  18. package/doc_assets/2.png +0 -0
  19. package/doc_assets/demo.md +27 -0
  20. package/doc_assets/demos/dist-app/assets/index.Dh-ZAS9Z.css +2 -0
  21. package/doc_assets/demos/dist-app/assets/index.Dv8KVW18.js +23699 -0
  22. package/doc_assets/demos/dist-app/assets/index.Dv8KVW18.js.map +1 -0
  23. package/doc_assets/demos/dist-app/index.html +14 -0
  24. package/doc_assets/demos/dist-app/vite.svg +1 -0
  25. package/doc_assets/images/1.png +0 -0
  26. package/doc_assets/images/3.png +0 -0
  27. package/doc_assets/images/component-screenshot.png +1 -0
  28. package/doc_assets/install.md +5 -0
  29. package/eslint.config.js +92 -0
  30. package/index.html +13 -0
  31. package/package.json +99 -44
  32. package/package.json.demo-backup +109 -0
  33. package/postcss.config.cjs +19 -0
  34. package/public/vite.svg +1 -0
  35. package/public/worker/pdf.worker.min.js +22 -0
  36. package/scripts/README.md +133 -0
  37. package/scripts/build-demo.js +88 -0
  38. package/scripts/demo-selector.js +216 -0
  39. package/scripts/dev-demo.js +76 -0
  40. package/scripts/preview-demo.js +130 -0
  41. package/scripts/run-demo.bat +34 -0
  42. package/src/assets/img/downArrow.png +0 -0
  43. package/src/assets/img/excel.png +0 -0
  44. package/src/assets/img/img.png +0 -0
  45. package/src/assets/img/pdf.png +0 -0
  46. package/src/assets/img/ppt.png +0 -0
  47. package/src/assets/img/txt.png +0 -0
  48. package/src/assets/img/word.png +0 -0
  49. package/src/assets/login/homeBg.png +0 -0
  50. package/src/assets/login/left.jpg +0 -0
  51. package/src/assets/login/logoImg.png +0 -0
  52. package/src/examples/LoginPage/index.tsx +18 -0
  53. package/src/examples/chat/components/DrawerGraphPreview.tsx +78 -0
  54. package/src/examples/chat/index.tsx +190 -0
  55. package/src/examples/gientechStreamFilesReader/index.tsx +950 -0
  56. package/src/examples/ragDatabaseDataPage/index.tsx +36 -0
  57. package/src/examples/ragDatabaseIdPage/index.tsx +44 -0
  58. package/src/examples/ragDatabasePage/index.tsx +36 -0
  59. package/src/examples/ragModelManagePage/index.tsx +37 -0
  60. package/src/examples/ragSearchPage/index.tsx +0 -0
  61. package/src/examples/ragSensitiveWordsPage/index.tsx +32 -0
  62. package/src/examples/streamFiles/index.tsx +384 -0
  63. package/src/lib_enter.ts +40 -0
  64. package/src/main.tsx +5 -0
  65. package/src/main.tsx.backup +5 -0
  66. package/src/modules/chat/Conversations/Item.tsx +167 -0
  67. package/src/modules/chat/Conversations/List.tsx +206 -0
  68. package/src/modules/chat/Conversations/groupByTime.ts +39 -0
  69. package/src/modules/chat/Conversations/index.tsx +217 -0
  70. package/src/modules/chat/constants.tsx +33 -0
  71. package/src/modules/chat/data.txt +82 -0
  72. package/src/modules/chat/index.tsx +1941 -0
  73. package/src/modules/chat/types.ts +17 -0
  74. package/src/modules/database/CreateModal.tsx +398 -0
  75. package/src/modules/database/assets/Doris.png +0 -0
  76. package/src/modules/database/assets/PostgreSQL.png +0 -0
  77. package/src/modules/database/assets/SQLServer.png +0 -0
  78. package/src/modules/database/assets/database.svg +11 -0
  79. package/src/modules/database/assets/database_add.svg +53 -0
  80. package/src/modules/database/assets/database_connect.svg +66 -0
  81. package/src/modules/database/assets/database_upload.svg +29 -0
  82. package/src/modules/database/assets/empty.png +0 -0
  83. package/src/modules/database/assets/mysql.svg +14 -0
  84. package/src/modules/database/index.tsx +466 -0
  85. package/src/modules/database/server.ts +196 -0
  86. package/src/modules/databaseId/CustomCom.tsx +156 -0
  87. package/src/modules/databaseId/EditConfig.tsx +268 -0
  88. package/src/modules/databaseId/UploadDrawer.tsx +520 -0
  89. package/src/modules/databaseId/assets/aiOptimize.svg +10 -0
  90. package/src/modules/databaseId/assets/empty.png +0 -0
  91. package/src/modules/databaseId/assets/template.svg +6 -0
  92. package/src/modules/databaseId/assets/upload.svg +9 -0
  93. package/src/modules/databaseId/assets/useTemp.svg +6 -0
  94. package/src/modules/databaseId/index.tsx +734 -0
  95. package/src/modules/databaseId/server.ts +286 -0
  96. package/src/modules/databaseId/style.css +5 -0
  97. package/src/modules/databaseTable/EditRowDrawer.tsx +119 -0
  98. package/src/modules/databaseTable/index.tsx +357 -0
  99. package/src/modules/databaseTable/server.ts +180 -0
  100. package/src/modules/headlessChat/constants.tsx +32 -0
  101. package/src/modules/headlessChat/index.tsx +1063 -0
  102. package/src/modules/headlessChat/types.ts +23 -0
  103. package/src/modules/login/components/Login/LoginBox/index.tsx +102 -0
  104. package/src/modules/login/components/Login/RegisterBox/index.tsx +180 -0
  105. package/src/modules/login/components/Login/index.tsx +100 -0
  106. package/src/modules/login/index.tsx +106 -0
  107. package/src/modules/login/style.css +3 -0
  108. package/src/modules/login/useServices.ts +53 -0
  109. package/src/modules/login/utils.ts +42 -0
  110. package/src/modules/modelManage/ConfigDrawer.tsx +249 -0
  111. package/src/modules/modelManage/assets/empty.png +0 -0
  112. package/src/modules/modelManage/const.ts +50 -0
  113. package/src/modules/modelManage/index.tsx +439 -0
  114. package/src/modules/modelManage/server.ts +185 -0
  115. package/src/modules/nodegraph/index.tsx +1 -0
  116. package/src/modules/search/assets/Icon-history.svg +8 -0
  117. package/src/modules/search/assets/answerAwartar.png +0 -0
  118. package/src/modules/search/assets/doc.png +0 -0
  119. package/src/modules/search/assets/genera.gif +0 -0
  120. package/src/modules/search/assets/icon-robot.svg +9 -0
  121. package/src/modules/search/assets/icon-search-bar.svg +14 -0
  122. package/src/modules/search/assets/icon-sub-title.svg +3 -0
  123. package/src/modules/search/assets/icon-title.svg +9 -0
  124. package/src/modules/search/assets/icon-zoomOut.svg +9 -0
  125. package/src/modules/search/assets/iconAi.svg +9 -0
  126. package/src/modules/search/assets/pdf.png +0 -0
  127. package/src/modules/search/assets/ppt.png +0 -0
  128. package/src/modules/search/assets/search.svg +3 -0
  129. package/src/modules/search/assets/selected.svg +4 -0
  130. package/src/modules/search/assets/txt.png +0 -0
  131. package/src/modules/search/assets/xls.png +0 -0
  132. package/src/modules/search/components/AssisSelect.tsx +137 -0
  133. package/src/modules/search/components/Editor/ChatViewEditor.tsx +261 -0
  134. package/src/modules/search/components/Editor/aichat.css +1 -0
  135. package/src/modules/search/components/Editor/constant.ts +13 -0
  136. package/src/modules/search/components/Editor/index.tsx +113 -0
  137. package/src/modules/search/components/Editor/plugins/autofomatRules.ts +332 -0
  138. package/src/modules/search/components/Editor/plugins/convertImgPlugins.tsx +20 -0
  139. package/src/modules/search/components/Editor/plugins/createIndexes.tsx +38 -0
  140. package/src/modules/search/components/Editor/plugins/displayer.ts +298 -0
  141. package/src/modules/search/components/Editor/plugins/imageClick.tsx +32 -0
  142. package/src/modules/search/components/Editor/plugins/myplugin.tsx +98 -0
  143. package/src/modules/search/components/Editor/ui/avatar.tsx +19 -0
  144. package/src/modules/search/components/Editor/ui/blockquote-element.tsx +21 -0
  145. package/src/modules/search/components/Editor/ui/button.tsx +58 -0
  146. package/src/modules/search/components/Editor/ui/calendar.tsx +68 -0
  147. package/src/modules/search/components/Editor/ui/caption.tsx +46 -0
  148. package/src/modules/search/components/Editor/ui/checkbox.tsx +27 -0
  149. package/src/modules/search/components/Editor/ui/code-block-combobox.tsx +188 -0
  150. package/src/modules/search/components/Editor/ui/code-block-element.css +434 -0
  151. package/src/modules/search/components/Editor/ui/code-block-element.tsx +39 -0
  152. package/src/modules/search/components/Editor/ui/code-leaf.tsx +24 -0
  153. package/src/modules/search/components/Editor/ui/code-line-element.tsx +10 -0
  154. package/src/modules/search/components/Editor/ui/code-syntax-leaf.tsx +21 -0
  155. package/src/modules/search/components/Editor/ui/column-element.tsx +30 -0
  156. package/src/modules/search/components/Editor/ui/column-group-element.tsx +94 -0
  157. package/src/modules/search/components/Editor/ui/command.tsx +75 -0
  158. package/src/modules/search/components/Editor/ui/comment-avatar.tsx +22 -0
  159. package/src/modules/search/components/Editor/ui/comment-create-form.tsx +37 -0
  160. package/src/modules/search/components/Editor/ui/comment-item.tsx +74 -0
  161. package/src/modules/search/components/Editor/ui/comment-leaf.tsx +49 -0
  162. package/src/modules/search/components/Editor/ui/comment-more-dropdown.tsx +42 -0
  163. package/src/modules/search/components/Editor/ui/comment-reply-items.tsx +22 -0
  164. package/src/modules/search/components/Editor/ui/comment-resolve-button.tsx +32 -0
  165. package/src/modules/search/components/Editor/ui/comment-value.tsx +34 -0
  166. package/src/modules/search/components/Editor/ui/comments-popover.tsx +63 -0
  167. package/src/modules/search/components/Editor/ui/date-element.tsx +83 -0
  168. package/src/modules/search/components/Editor/ui/dialog.tsx +63 -0
  169. package/src/modules/search/components/Editor/ui/draggable.tsx +177 -0
  170. package/src/modules/search/components/Editor/ui/dropdown-menu.tsx +180 -0
  171. package/src/modules/search/components/Editor/ui/emoji-input-element.tsx +85 -0
  172. package/src/modules/search/components/Editor/ui/excalidraw-element.tsx +28 -0
  173. package/src/modules/search/components/Editor/ui/fixed-toolbar-buttons.tsx +76 -0
  174. package/src/modules/search/components/Editor/ui/fixed-toolbar.tsx +8 -0
  175. package/src/modules/search/components/Editor/ui/floating-toolbar-buttons.tsx +51 -0
  176. package/src/modules/search/components/Editor/ui/floating-toolbar.tsx +77 -0
  177. package/src/modules/search/components/Editor/ui/heading-element.tsx +48 -0
  178. package/src/modules/search/components/Editor/ui/highlight-leaf.tsx +17 -0
  179. package/src/modules/search/components/Editor/ui/hr-element.tsx +30 -0
  180. package/src/modules/search/components/Editor/ui/icons.tsx +267 -0
  181. package/src/modules/search/components/Editor/ui/image-element.tsx +74 -0
  182. package/src/modules/search/components/Editor/ui/inline-combobox.tsx +368 -0
  183. package/src/modules/search/components/Editor/ui/input.tsx +25 -0
  184. package/src/modules/search/components/Editor/ui/insert-dropdown-menu.tsx +218 -0
  185. package/src/modules/search/components/Editor/ui/kbd-leaf.tsx +20 -0
  186. package/src/modules/search/components/Editor/ui/link-element.tsx +29 -0
  187. package/src/modules/search/components/Editor/ui/link-floating-toolbar.tsx +161 -0
  188. package/src/modules/search/components/Editor/ui/list-element.tsx +30 -0
  189. package/src/modules/search/components/Editor/ui/mark-toolbar-button.tsx +24 -0
  190. package/src/modules/search/components/Editor/ui/media-embed-element.tsx +133 -0
  191. package/src/modules/search/components/Editor/ui/media-popover.tsx +97 -0
  192. package/src/modules/search/components/Editor/ui/mention-element.tsx +43 -0
  193. package/src/modules/search/components/Editor/ui/mention-input-element.tsx +141 -0
  194. package/src/modules/search/components/Editor/ui/mode-dropdown-menu.tsx +93 -0
  195. package/src/modules/search/components/Editor/ui/more-dropdown-menu.tsx +67 -0
  196. package/src/modules/search/components/Editor/ui/paragraph-element.tsx +4 -0
  197. package/src/modules/search/components/Editor/ui/placeholder.tsx +52 -0
  198. package/src/modules/search/components/Editor/ui/popover.tsx +32 -0
  199. package/src/modules/search/components/Editor/ui/resizable.tsx +66 -0
  200. package/src/modules/search/components/Editor/ui/separator.tsx +25 -0
  201. package/src/modules/search/components/Editor/ui/style.less +12 -0
  202. package/src/modules/search/components/Editor/ui/table-cell-element.tsx +143 -0
  203. package/src/modules/search/components/Editor/ui/table-element.tsx +243 -0
  204. package/src/modules/search/components/Editor/ui/table-row-element.tsx +22 -0
  205. package/src/modules/search/components/Editor/ui/tableValue.tsx +135 -0
  206. package/src/modules/search/components/Editor/ui/todo-list-element.tsx +43 -0
  207. package/src/modules/search/components/Editor/ui/toggle-element.tsx +31 -0
  208. package/src/modules/search/components/Editor/ui/toolbar.tsx +157 -0
  209. package/src/modules/search/components/Editor/ui/tooltip.tsx +65 -0
  210. package/src/modules/search/components/Editor/ui/turn-into-dropdown-menu.tsx +160 -0
  211. package/src/modules/search/components/Editor/ui/with-draggables.tsx +175 -0
  212. package/src/modules/search/components/FileList.tsx +287 -0
  213. package/src/modules/search/components/ImageGroupView/index.tsx +85 -0
  214. package/src/modules/search/components/ResultContent.tsx +232 -0
  215. package/src/modules/search/components/SearchInput.tsx +232 -0
  216. package/src/modules/search/components/SearchLanding.tsx +74 -0
  217. package/src/modules/search/components/SearchView.tsx +563 -0
  218. package/src/modules/search/components/SimpleEditor.tsx +158 -0
  219. package/src/modules/search/components/SimpleFileList.tsx +215 -0
  220. package/src/modules/search/index.tsx +10 -0
  221. package/src/modules/search/reademe.md +1 -0
  222. package/src/modules/search/servers/apis.tsx +19 -0
  223. package/src/modules/search/servers/index.ts +184 -0
  224. package/src/modules/search/style.less +503 -0
  225. package/src/modules/search/type.ts +22 -0
  226. package/src/modules/search/utils.ts +34 -0
  227. package/src/modules/sensitive/index.tsx +313 -0
  228. package/src/modules/sensitive/server.ts +122 -0
  229. package/src/modules/streamFilesReader/GientechStreamReader.tsx +1619 -0
  230. package/src/modules/streamFilesReader/components/Header/Toolbar.tsx +0 -0
  231. package/src/modules/streamFilesReader/components/Header/index.tsx +297 -0
  232. package/src/modules/streamFilesReader/index.tsx +3 -0
  233. package/src/style.css +6 -0
  234. package/src/type.d.ts +0 -0
  235. package/src/utils/commonFn.tsx +111 -0
  236. package/src/utils/gientechCommon/components/AppError.tsx +32 -0
  237. package/src/utils/gientechCommon/components/AppLoading.tsx +75 -0
  238. package/src/utils/gientechCommon/components/DeleteModal.tsx +75 -0
  239. package/src/utils/gientechCommon/components/DisplayError.tsx +33 -0
  240. package/src/utils/gientechCommon/components/DisplayLoading.tsx +38 -0
  241. package/src/utils/gientechCommon/components/FeedBackModal.tsx +310 -0
  242. package/src/utils/gientechCommon/components/FileCardCommon.tsx +82 -0
  243. package/src/utils/gientechCommon/components/FileManager/index.tsx +390 -0
  244. package/src/utils/gientechCommon/components/FileManager/style.css +5 -0
  245. package/src/utils/gientechCommon/components/Messages/GientechNewChatWelcome.tsx +296 -0
  246. package/src/utils/gientechCommon/components/Messages/ReferenceCard.tsx +339 -0
  247. package/src/utils/gientechCommon/components/Messages/RetriveItem.tsx +245 -0
  248. package/src/utils/gientechCommon/components/Messages/WebRetriveItem.tsx +209 -0
  249. package/src/utils/gientechCommon/components/Messages/defaultBot.png +0 -0
  250. package/src/utils/gientechCommon/components/Messages/defaultStyleSet.tsx +148 -0
  251. package/src/utils/gientechCommon/components/Messages/defaultWeLogo.svg +14 -0
  252. package/src/utils/gientechCommon/components/RenameModal.tsx +86 -0
  253. package/src/utils/gientechCommon/components/style.less +11 -0
  254. package/src/utils/gientechCommon/configs/commonConfig.ts +2 -0
  255. package/src/utils/gientechCommon/configs/senderConfig.ts +0 -0
  256. package/src/utils/gientechCommon/configs/stylesConfig.ts +142 -0
  257. package/src/utils/gientechCommon/hooks/AichatUseController.tsx +345 -0
  258. package/src/utils/gientechCommon/slate/converters/deserializers.ts +763 -0
  259. package/src/utils/gientechCommon/slate/converters/mockData.ts +232 -0
  260. package/src/utils/gientechCommon/slate/converters/slateConverters.ts +258 -0
  261. package/src/utils/gientechCommon/slate/richElements/index.tsx +499 -0
  262. package/src/utils/gientechCommon/utils/request.ts +37 -0
  263. package/src/utils/gientechCommon/utils/serverFn.ts +172 -0
  264. package/src/utils/index.tsx +126 -0
  265. package/src/utils/testconfigs/demologin/index.tsx +32 -0
  266. package/src/utils/testconfigs/index.ts +53 -0
  267. package/src/vite-env.d.ts +11 -0
  268. package/stats.html +4949 -0
  269. package/tailwind.config.js +170 -0
  270. package/tsconfig.app.json +30 -0
  271. package/tsconfig.app.tsbuildinfo +11 -0
  272. package/tsconfig.json +13 -0
  273. package/tsconfig.node.json +22 -0
  274. package/tsconfig.node.tsbuildinfo +1 -0
  275. package/vite.config.ts +216 -0
  276. package/workflows/release.yml +60 -0
  277. package/database.js +0 -20
  278. /package/{assets → dist/assets}/AppLoading-BCu9TPs5.js +0 -0
  279. /package/{assets → dist/assets}/Doris.png +0 -0
  280. /package/{assets → dist/assets}/GientechStreamReader-A8zHypyu.js +0 -0
  281. /package/{assets → dist/assets}/LeftOutlined-BdBZ6FSD.js +0 -0
  282. /package/{assets → dist/assets}/PostgreSQL.png +0 -0
  283. /package/{assets → dist/assets}/SQLServer.png +0 -0
  284. /package/{assets → dist/assets}/empty.png +0 -0
  285. /package/{assets → dist/assets}/homeBg.png +0 -0
  286. /package/{assets → dist/assets}/index-B7u9FawR.js +0 -0
  287. /package/{assets → dist/assets}/index-CFlRqq2H.js +0 -0
  288. /package/{assets → dist/assets}/index-CNJZ9j4J.js +0 -0
  289. /package/{assets → dist/assets}/index-CpW6Dhpp.js +0 -0
  290. /package/{assets → dist/assets}/index-DfSHCQcq.js +0 -0
  291. /package/{assets → dist/assets}/index-DiQ4Me9J.js +0 -0
  292. /package/{assets → dist/assets}/index-EcdpegTC.js +0 -0
  293. /package/{assets → dist/assets}/index-rydP_SDR.js +0 -0
  294. /package/{assets → dist/assets}/left.jpg +0 -0
  295. /package/{assets → dist/assets}/logoImg.png +0 -0
  296. /package/{assets → dist/assets}/plus-ufD39rmY.js +0 -0
  297. /package/{assets → dist/assets}/style.css +0 -0
  298. /package/{assets → dist/assets}/style2.css +0 -0
  299. /package/{assets → dist/assets}/style3.css +0 -0
  300. /package/{assets → dist/assets}/style4.css +0 -0
  301. /package/{assets → dist/assets}/x-_1-UpSBt.js +0 -0
  302. /package/{chat.d.ts → dist/chat.d.ts} +0 -0
  303. /package/{chat.js → dist/chat.js} +0 -0
  304. /package/{database.d.ts → dist/database.d.ts} +0 -0
  305. /package/{databaseId.d.ts → dist/databaseId.d.ts} +0 -0
  306. /package/{databaseId.js → dist/databaseId.js} +0 -0
  307. /package/{databaseTable.d.ts → dist/databaseTable.d.ts} +0 -0
  308. /package/{databaseTable.js → dist/databaseTable.js} +0 -0
  309. /package/{index.d.ts → dist/index.d.ts} +0 -0
  310. /package/{index.js → dist/index.js} +0 -0
  311. /package/{modelManage.d.ts → dist/modelManage.d.ts} +0 -0
  312. /package/{modelManage.js → dist/modelManage.js} +0 -0
  313. /package/{sensitive.d.ts → dist/sensitive.d.ts} +0 -0
  314. /package/{sensitive.js → dist/sensitive.js} +0 -0
  315. /package/{streamFilesReader.d.ts → dist/streamFilesReader.d.ts} +0 -0
  316. /package/{streamFilesReader.js → dist/streamFilesReader.js} +0 -0
  317. /package/{vite.svg → dist/vite.svg} +0 -0
@@ -0,0 +1,167 @@
1
+ import React, { useMemo, useState } from 'react';
2
+ import { Dropdown } from 'antd';
3
+ import type { MenuProps } from 'antd';
4
+ import { MoreHorizontal, Edit2, Trash2 } from 'lucide-react';
5
+ import { ListItemType } from '.';
6
+ import dayjs from 'dayjs';
7
+
8
+ export interface GientechConversationBarItemProps {
9
+ isActive: boolean;
10
+ isDisabled: boolean;
11
+ data: ListItemType;
12
+ isNew: boolean;
13
+ eventsEmit?: (type: string, data?: any) => void;
14
+ styles?: any;
15
+ assistantList?: any[];
16
+ }
17
+
18
+ export function GientechConversationBarItem({ isActive, isNew, isDisabled, data, eventsEmit, styles = {}, assistantList = [] }: GientechConversationBarItemProps) {
19
+ const [hovered, setHovered] = useState(false);
20
+ const colors = styles?.colors || {};
21
+
22
+ // 由 configId 匹配助手名称(优先用 name,其次 configJson.config_name)
23
+ const assistantName = useMemo(() => {
24
+ const id = data?.configId;
25
+ if (!id || !assistantList || assistantList.length === 0) return '未选择助手';
26
+ const found = assistantList.find((a: any) => String(a?.id) === String(id));
27
+ if (!found) return '未选择助手';
28
+ if (found?.name) return found.name;
29
+ try {
30
+ const cj = typeof found.configJson === 'string' ? JSON.parse(found.configJson) : found.configJson;
31
+ return cj?.config_name || '未选择助手';
32
+ } catch {
33
+ return '未选择助手';
34
+ }
35
+ }, [assistantList, data?.configId]);
36
+
37
+ // antd v5+ menu items
38
+ const menuItems: MenuProps['items'] = [
39
+ {
40
+ key: 'rename',
41
+ icon: <Edit2 size={12} />,
42
+ label: '重命名',
43
+ onClick: (info) => {
44
+ info.domEvent.stopPropagation();
45
+ eventsEmit&&eventsEmit('conversations:rename_icon_clicked', data);
46
+ },
47
+ },
48
+ {
49
+ key: 'delete',
50
+ icon: <Trash2 size={12} />,
51
+ label: '删除',
52
+ danger: true,
53
+ onClick: (info) => {
54
+ info.domEvent.stopPropagation();
55
+ eventsEmit&&eventsEmit('conversations:delete_icon_clicked', data);
56
+ },
57
+ },
58
+ ];
59
+
60
+
61
+ // 动态样式
62
+ const itemBg = isActive
63
+ ? colors.activeItemBg || '#f0f4ff'
64
+ : hovered
65
+ ? colors.itemHoverBg || '#f5f7fa'
66
+ : colors.itemBg || 'transparent';
67
+ const itemText = isActive
68
+ ? colors.activeItemText || colors.primary || '#0122ff'
69
+ : colors.itemText || '#333';
70
+ const borderColor = isActive
71
+ ? colors.activeItemBorder || colors.primary || '#0122ff'
72
+ : 'transparent';
73
+
74
+ // 格式化时间显示
75
+ const formatTime = (timestamp: number) => {
76
+ const now = dayjs();
77
+ const messageTime = dayjs(timestamp);
78
+ const isToday = now.isSame(messageTime, 'day');
79
+
80
+ if (isToday) {
81
+ return messageTime.format('HH:mm');
82
+ } else {
83
+ return messageTime.format('MM-DD HH:mm');
84
+ }
85
+ };
86
+
87
+ return (
88
+ <div
89
+ className={`flex relative items-center justify-between mb-2 px-4 py-3 rounded-md transition-colors cursor-pointer select-none`}
90
+ onClick={() => !isDisabled && eventsEmit&&eventsEmit('conversation:item_click', data)}
91
+ onMouseEnter={() => setHovered(true)}
92
+ onMouseLeave={() => setHovered(false)}
93
+ style={{
94
+ minHeight: 48,
95
+ background: itemBg,
96
+ color: itemText,
97
+ border: `1px solid ${borderColor}`,
98
+ ...styles.item,
99
+ opacity: isDisabled ? 0.5 : 1,
100
+ }}
101
+ >
102
+ <div className="flex flex-col gap-y-2 flex-1 min-w-0">
103
+ {/* 主标题 */}
104
+ <span
105
+ className="truncate text-sm max-w-[160px] font-medium"
106
+ title={data.label || '未命名会话'}
107
+ style={{
108
+ color: itemText,
109
+ fontSize: '14px',
110
+ lineHeight: '1.4'
111
+ }}
112
+ >
113
+ {data.label || '未命名会话'}
114
+ </span>
115
+
116
+ {/* 副标题和时间 */}
117
+ <div className="flex items-center justify-between gap-2">
118
+ <span
119
+ className="text-xs truncate flex-1"
120
+ style={{
121
+ color: isActive ? colors.activeItemTextSecondary || '#666' : colors.itemTextSecondary || '#666',
122
+ fontSize: '12px',
123
+ lineHeight: '1.3'
124
+ }}
125
+ >
126
+ {assistantName}
127
+ </span>
128
+
129
+ {/* 时间显示 */}
130
+ {data.gmtModified && (
131
+ <span
132
+ className="text-xs flex-shrink-0"
133
+ style={{
134
+ color: isActive ? colors.activeItemTextSecondary || '#999' : colors.itemTextSecondary || '#999',
135
+ fontSize: '12px',
136
+ lineHeight: '1.3'
137
+ }}
138
+ >
139
+ {formatTime(data.gmtModified)}
140
+ </span>
141
+ )}
142
+ </div>
143
+ </div>
144
+
145
+ {/* 省略号菜单 */}
146
+ {!isDisabled && hovered && !isNew && (
147
+ <div className="absolute top-2 right-3">
148
+ <Dropdown
149
+ menu={{ items: menuItems }}
150
+ trigger={['click']}
151
+ placement="bottomRight"
152
+ >
153
+ <MoreHorizontal
154
+ size={16}
155
+ className="text-gray-600 hover:text-gray-800"
156
+ style={{
157
+ cursor: 'pointer',
158
+ color: isActive ? colors.activeItemTextSecondary || '#999' : colors.itemTextSecondary || '#999'
159
+ }}
160
+ onClick={(e: React.MouseEvent) => e.stopPropagation()}
161
+ />
162
+ </Dropdown>
163
+ </div>
164
+ )}
165
+ </div>
166
+ );
167
+ }
@@ -0,0 +1,206 @@
1
+ import React, { useMemo, useRef, useEffect } from 'react';
2
+ import { VariableSizeList as List } from 'react-window';
3
+ import { GientechConversationBarItem } from './Item';
4
+ import { groupConversationsByTime, ListItemType } from './groupByTime';
5
+ import { Flipper, Flipped } from 'react-flip-toolkit';
6
+
7
+ export interface GientechConversationBarListProps {
8
+ data: ListItemType[];
9
+ activedItem: string;
10
+ onChange?: (sessionId: string) => void;
11
+ onDelete?: (item: ListItemType) => void;
12
+ sortRules?: { label: string; dayPeriod: number }[];
13
+ eventsEmit?: (eventName: string, data?: any) => void;
14
+ assistantList?: any[];
15
+ height?: number;
16
+ styles?: any;
17
+ hasMore?: boolean;
18
+ loadingMore?: boolean;
19
+ }
20
+
21
+ export function GientechConversationBarList(
22
+ props: GientechConversationBarListProps & { styles?: any }
23
+ ) {
24
+ const {
25
+ data,
26
+ activedItem,
27
+ onChange,
28
+ onDelete,
29
+ sortRules = [{ label: '当天', dayPeriod: 1 }],
30
+ eventsEmit,
31
+ assistantList,
32
+ height = 400,
33
+ styles = {},
34
+ hasMore = false,
35
+ loadingMore = false,
36
+ } = props;
37
+
38
+ // 用于动画的 flipKey,包含 sessionId 和 gmtModified
39
+ const flipKey = data.map(i => i.sessionId + '-' + i.gmtModified).join(',');
40
+
41
+ const flatList = useMemo(() => {
42
+ const rules = sortRules.length > 0 ? sortRules : [{ label: '当天', dayPeriod: 1 }];
43
+ const sortedList = groupConversationsByTime(data, rules);
44
+ const hasOtherOnly = rules.every(
45
+ rule => !sortedList[rule.label] || sortedList[rule.label].length === 0
46
+ );
47
+ const arr: { type: 'group' | 'item'; label?: string; data?: ListItemType }[] = [];
48
+ rules.forEach(rule => {
49
+ if (sortedList[rule.label] && sortedList[rule.label].length > 0) {
50
+ arr.push({ type: 'group', label: rule.label });
51
+ sortedList[rule.label].forEach(i => arr.push({ type: 'item', data: i }));
52
+ }
53
+ });
54
+ if (sortedList.other.length > 0 && !hasOtherOnly) {
55
+ arr.push({ type: 'group', label: '其他' });
56
+ sortedList.other.forEach(i => arr.push({ type: 'item', data: i }));
57
+ }
58
+ if (hasOtherOnly && sortedList.other.length > 0) {
59
+ sortedList.other.forEach(i => arr.push({ type: 'item', data: i }));
60
+ }
61
+ return arr;
62
+ }, [data, sortRules, flipKey]);
63
+
64
+ // 代表当前扁平顺序与类型的签名,用于监测重排
65
+ const layoutKey = useMemo(() => {
66
+ return flatList
67
+ .map(i => (i.type === 'group' ? `g:${i.label}` : `i:${i.data!.sessionId}`))
68
+ .join('|');
69
+ }, [flatList]);
70
+
71
+ const ITEM_HEIGHT = 56;
72
+ const ITEM_GAP = 14;
73
+ const FOOTER_HEIGHT = 36;
74
+
75
+ const getItemSize = (index: number) => {
76
+ // 最后一项为 Footer
77
+ if (index === flatList.length) return FOOTER_HEIGHT;
78
+ return flatList[index].type === 'group' ? 32 : ITEM_HEIGHT + ITEM_GAP;
79
+ };
80
+
81
+ // 新增:List ref
82
+ const listRef = useRef<any>(null);
83
+ // 新增:当前激活会话在 flatList 的 index(仅依赖 activedItem 与数据映射)
84
+ const activeIndex = useMemo(() => {
85
+ return flatList.findIndex(item => item.type === 'item' && item.data?.sessionId === activedItem);
86
+ }, [flatList, activedItem]);
87
+ // 新增:flatList 里第一个会话项的 index
88
+ const firstItemIndex = useMemo(() => {
89
+ return flatList.findIndex(item => item.type === 'item');
90
+ }, [flatList]);
91
+ // 仅在激活项变化时,且激活项恰好是第一项时滚动到顶部;加载更多时不触发
92
+ const lastActivedRef = useRef<string | undefined>(undefined);
93
+ useEffect(() => {
94
+ const last = lastActivedRef.current;
95
+ if (activedItem !== last && activeIndex > -1 && activeIndex === firstItemIndex && listRef.current) {
96
+ listRef.current.scrollToItem(activeIndex, 'start');
97
+ }
98
+ lastActivedRef.current = activedItem;
99
+ }, [activedItem, activeIndex, firstItemIndex]);
100
+
101
+ // 轻量刷新缓存:仅在数据长度变化时刷新末尾,避免重置滚动位置
102
+ const prevCountRef = useRef<number>(0);
103
+ useEffect(() => {
104
+ if (listRef.current) {
105
+ const prev = prevCountRef.current;
106
+ const curr = flatList.length;
107
+ prevCountRef.current = curr;
108
+ if (curr !== prev) {
109
+ listRef.current.resetAfterIndex(Math.max(0, Math.min(prev, curr) - 1), false);
110
+ }
111
+ }
112
+ }, [flatList.length]);
113
+
114
+ // 当分组/顺序发生变化时,重置尺寸缓存,避免位置叠加
115
+ useEffect(() => {
116
+ if (listRef.current) {
117
+ listRef.current.resetAfterIndex(0, false);
118
+ }
119
+ }, [layoutKey]);
120
+
121
+ // hooks全部调用完后再判断是否为空
122
+ if (flatList.length === 0) {
123
+ return <div className="text-center text-zinc-400 py-8">暂无会话</div>;
124
+ }
125
+
126
+ const colors = styles?.colors || {};
127
+
128
+ return (
129
+ <Flipper flipKey={flipKey} spring={{ stiffness: 180, damping: 18 }}>
130
+ <List
131
+ ref={listRef}
132
+ height={height}
133
+ width={'100%'}
134
+
135
+ itemCount={flatList.length + 1}
136
+ itemSize={getItemSize}
137
+ itemKey={index => {
138
+ if (index === flatList.length) return 'footer';
139
+ const item = flatList[index];
140
+ return item.type === 'item' ? item.data!.sessionId : `group-${item.label}`;
141
+ }}
142
+ onItemsRendered={({ visibleStopIndex }) => {
143
+ // 接近底部自动加载
144
+ const threshold = 3;
145
+ if (
146
+ hasMore &&
147
+ !loadingMore &&
148
+ visibleStopIndex >= flatList.length - 1 - threshold &&
149
+ typeof eventsEmit === 'function'
150
+ ) {
151
+ eventsEmit('conversations:load_more');
152
+ }
153
+ }}
154
+ style={{ overflowX: 'hidden' }}
155
+ >
156
+ {({ index, style }) => {
157
+ // Footer 行
158
+ if (index === flatList.length) {
159
+ const colors = styles?.colors || {};
160
+ return (
161
+ <div
162
+ style={{
163
+ ...style,
164
+ height: FOOTER_HEIGHT,
165
+ lineHeight: `${FOOTER_HEIGHT}px`,
166
+ textAlign: 'center',
167
+ fontSize: '10px',
168
+ color: colors.textMuted || '#9ca3af',
169
+ background: colors.groupBg || 'white',
170
+ }}
171
+ >
172
+ {loadingMore ? '加载中...' : hasMore ? '' : '没有更多了'}
173
+ </div>
174
+ );
175
+ }
176
+ const item = flatList[index];
177
+ if (item.type === 'group') {
178
+ return (
179
+ <div
180
+ style={{ ...style, background: colors.groupBg || 'white', zIndex: 1 }}
181
+ className="pt-2 pb-1 px-[28px] py-[2px] text-xs text-gray-400"
182
+ >
183
+ {item.label}
184
+ </div>
185
+ );
186
+ }
187
+ return (
188
+ <Flipped key={item.data!.sessionId} flipId={item.data!.sessionId}>
189
+ <div style={style} className='px-[12px] py-[2px]'>
190
+ <GientechConversationBarItem
191
+ isActive={activedItem === item.data!.sessionId}
192
+ isDisabled={false}
193
+ isNew={item.data!.isNew}
194
+ data={item.data!}
195
+ eventsEmit={eventsEmit}
196
+ assistantList={assistantList}
197
+ styles={styles}
198
+ />
199
+ </div>
200
+ </Flipped>
201
+ );
202
+ }}
203
+ </List>
204
+ </Flipper>
205
+ );
206
+ }
@@ -0,0 +1,39 @@
1
+ export interface ListItemType {
2
+ label: string;
3
+ sessionId: string;
4
+ gmtModified: number;
5
+ [key: string]: any;
6
+ }
7
+ export interface SortRule {
8
+ label: string;
9
+ dayPeriod: number;
10
+ }
11
+
12
+ /**
13
+ * 按时间分组会话数据
14
+ * @param data 会话列表
15
+ * @param rules 分组规则
16
+ * @returns 分组后的对象
17
+ */
18
+ export function groupConversationsByTime(
19
+ data: ListItemType[],
20
+ rules: SortRule[] = [{ label: '当天', dayPeriod: 1 }]
21
+ ): Record<string, ListItemType[]> {
22
+ const now = new Date();
23
+ const sortedList: Record<string, ListItemType[]> = { other: [] };
24
+ data.forEach(item => {
25
+ const itemDate = new Date(item.gmtModified);
26
+ const daysDifference = Math.floor((+now - +itemDate) / (1000 * 60 * 60 * 24));
27
+ let found = false;
28
+ for (const rule of rules) {
29
+ if (daysDifference <= rule.dayPeriod) {
30
+ if (!sortedList[rule.label]) sortedList[rule.label] = [];
31
+ sortedList[rule.label].push(item);
32
+ found = true;
33
+ break;
34
+ }
35
+ }
36
+ if (!found) sortedList.other.push(item);
37
+ });
38
+ return sortedList;
39
+ }
@@ -0,0 +1,217 @@
1
+ import React, { useState, useMemo, useEffect, useRef } from 'react';
2
+
3
+ import { GientechConversationBarList } from './List';
4
+ import { Plus, Search, X } from 'lucide-react';
5
+ export interface ListItemType {
6
+ label: string;
7
+ sessionId: string;
8
+ gmtModified: number;
9
+ [key: string]: any;
10
+ }
11
+ export interface GientechConversationPanelProps {
12
+ data: ListItemType[];
13
+ sortRules?: { label: string; dayPeriod: number }[];
14
+ activedItem: string;
15
+ eventsEmit?: (eventName: string, data?: any) => void;
16
+ assistantList?: any[];
17
+ styles?: any; // 新增,支持动态主题样式
18
+ hasMore?: boolean;
19
+ loadingMore?: boolean;
20
+ }
21
+
22
+ // Header美化:分区更明显,搜索框更圆润,按钮更突出,底部分割线阴影
23
+ const Header: React.FC<{
24
+ value: string;
25
+ setSearch: (v: string) => void;
26
+ eventsEmit?: (eventName: string, data?: any) => void;
27
+ style?: React.CSSProperties;
28
+ styles?: any;
29
+ }> = ({ value, eventsEmit, setSearch, style, styles: themeStyles = {} }) => {
30
+ const colors = themeStyles.colors || {};
31
+ const radius = themeStyles.radius || '12px';
32
+ const border = colors.border || '#e5e7eb';
33
+ const primary = colors.primary || '#2563eb';
34
+ const headerBg = themeStyles.headerBg || colors.sidebarBg || colors.background || '#fff';
35
+ const [searchOpen, setSearchOpen] = React.useState(false);
36
+ return (
37
+ <div
38
+ className="flex items-center gap-2 px-4 py-3 sticky top-0 z-10 min-w-0"
39
+ style={{
40
+ ...style,
41
+ background: headerBg,
42
+ // borderBottom: `1px solid ${border}`,
43
+ borderTopLeftRadius: radius,
44
+ borderTopRightRadius: radius,
45
+ }}
46
+ >
47
+ {/* 新建按钮:通过宽度/透明度动画隐藏,保持DOM稳定 */}
48
+ <button
49
+ className="border-none flex items-center justify-center"
50
+ onClick={() => {
51
+ // 新建会话后保持默认布局
52
+ setSearchOpen(false);
53
+ eventsEmit && eventsEmit('conversation:create');
54
+ }}
55
+ title="新建会话"
56
+ style={{
57
+ height: 42,
58
+ borderRadius: '90px',
59
+ background: primary,
60
+ color: colors.buttonText || '#fff',
61
+ fontWeight: 600,
62
+ boxShadow: themeStyles.shadow || '0 2px 8px 0 rgba(0,0,0,0.04)',
63
+ padding: searchOpen ? '0 0' : '0 14px',
64
+ minWidth: searchOpen ? 0 : 180,
65
+ width: searchOpen ? 0 : undefined,
66
+ opacity: searchOpen ? 0 : 1,
67
+ overflow: 'hidden',
68
+ flexShrink: 0,
69
+ transition: 'width .22s ease, min-width .22s ease, padding .22s ease, opacity .18s ease',
70
+ }}
71
+ >
72
+ <div className="flex items-center gap-2">
73
+ <Plus size={18} />
74
+ <span className="text-sm">新建对话</span>
75
+ </div>
76
+ </button>
77
+
78
+ {/* 搜索区:折叠图标/展开输入框 */}
79
+ {!searchOpen ? (
80
+ <button
81
+ className="ml-auto flex items-center justify-center transition-colors"
82
+ style={{
83
+ width: 36,
84
+ height: 36,
85
+ borderRadius: '90px',
86
+ // background: colors.inputBg || '#f4f4f5',
87
+ border: `1px solid ${border}`,
88
+ color: primary || '#222',
89
+ flexShrink: 0,
90
+ }}
91
+ onClick={() => setSearchOpen(true)}
92
+ title="搜索"
93
+ >
94
+ <Search size={18} />
95
+ </button>
96
+ ) : (
97
+ <div className="flex-1 min-w-0 flex items-center gap-2">
98
+ <input
99
+ className="flex-1 px-3 py-2 text-sm transition-all"
100
+ placeholder="搜索会话..."
101
+ value={value}
102
+ onChange={e => setSearch(e.target.value)}
103
+ onBlur={() => setSearchOpen(false)}
104
+ onKeyDown={e => {
105
+ if (e.key === 'Enter') {
106
+ setSearchOpen(false);
107
+ }
108
+ }}
109
+ style={{
110
+ minWidth: 0,
111
+ width: '100%',
112
+ maxWidth: '100%',
113
+ borderRadius: radius,
114
+ background: colors.inputBg || '#f4f4f5',
115
+ border: `1px solid ${border}`,
116
+ color: colors.text || '#222',
117
+ outline: 'none',
118
+ }}
119
+ />
120
+ <button
121
+ className="flex items-center justify-center transition-colors"
122
+ style={{
123
+ width: 36,
124
+ height: 36,
125
+ borderRadius: '90px',
126
+ // background: colors.inputBg || '#f4f4f5',
127
+ border: `1px solid ${border}`,
128
+ color: colors.text || '#222',
129
+ flexShrink: 0,
130
+ }}
131
+ onClick={() => setSearchOpen(false)}
132
+ title="收起搜索"
133
+ >
134
+ <X size={16} />
135
+ </button>
136
+ </div>
137
+ )}
138
+ </div>
139
+ );
140
+ };
141
+
142
+ export default function GientechConversationPanel(props: GientechConversationPanelProps) {
143
+ const { data, sortRules, activedItem,eventsEmit,assistantList, styles = {}, hasMore = false, loadingMore = false } = props;
144
+ const [search, setSearch] = useState('');
145
+
146
+ // 本地搜索过滤
147
+ const filteredData = useMemo(() => {
148
+ if (!search) return data;
149
+ return data.filter(item => (item.label || '未命名会话').toLowerCase().includes(search.toLowerCase()));
150
+ }, [data, search]);
151
+
152
+ // 新增:对 filteredData 按 gmtModified 倒序排序
153
+ const sortedData = useMemo(() => {
154
+ return [...filteredData].sort((a, b) => new Date(b.gmtModified).getTime() - new Date(a.gmtModified).getTime());
155
+ }, [filteredData]);
156
+
157
+ // 动态获取会话列表容器高度
158
+ const listContainerRef = useRef<HTMLDivElement>(null);
159
+ const [listHeight, setListHeight] = useState(400);
160
+ useEffect(() => {
161
+ function updateHeight() {
162
+ if (listContainerRef.current) {
163
+ setListHeight(listContainerRef.current.clientHeight);
164
+ }
165
+ }
166
+ updateHeight();
167
+ window.addEventListener('resize', updateHeight);
168
+ return () => window.removeEventListener('resize', updateHeight);
169
+ }, []);
170
+
171
+ // 主题色提取
172
+ const colors = styles?.colors || {};
173
+ const radius = styles?.radius || '12px';
174
+ const shadow = styles?.shadow || '0 4px 24px 0 rgba(0,0,0,0.06)';
175
+ const border = colors.border || '#e5e7eb';
176
+ const sidebarBg = colors.sidebarBg || colors.background || '#EAEEFF';
177
+ const textColor = colors.text || '#222';
178
+
179
+ return (
180
+ <div
181
+ className="h-full flex flex-col overflow-y-hidden"
182
+ style={{
183
+ background: sidebarBg,
184
+ color: textColor,
185
+ // borderRadius: radius,
186
+ boxShadow: shadow,
187
+ border: `1px solid ${border}`,
188
+ ...styles.container,
189
+ }}
190
+ >
191
+ {/* Header 区域美化 */}
192
+ <Header value={search} setSearch={setSearch} eventsEmit={eventsEmit} style={styles.header} styles={styles} />
193
+ {/* 会话列表区域,内容溢出时顶部有阴影提示 */}
194
+ <div className="flex-1 min-h-0 relative" ref={listContainerRef} style={styles.list}>
195
+ <GientechConversationBarList
196
+ eventsEmit={eventsEmit}
197
+ data={sortedData}
198
+ assistantList={assistantList}
199
+ activedItem={activedItem}
200
+ sortRules={sortRules}
201
+ height={listHeight}
202
+ styles={styles}
203
+ hasMore={hasMore}
204
+ loadingMore={loadingMore}
205
+ />
206
+ {/* 顶部阴影提示 */}
207
+ <div
208
+ className="pointer-events-none absolute left-0 right-0 top-0 h-3 z-20"
209
+ style={{
210
+ background: `linear-gradient(to bottom, ${sidebarBg}cc, transparent)`
211
+ }}
212
+ />
213
+ </div>
214
+ {/* Footer 文案改由虚拟列表内部渲染,这里无需再显示 */}
215
+ </div>
216
+ );
217
+ }
@@ -0,0 +1,33 @@
1
+ import { Icon } from '@mxmweb/zui';
2
+
3
+ export const DefaultSenderConfig = {
4
+ actions: [
5
+ {
6
+ icon: <Icon type='rag/folder' size={18} />,
7
+ name: 'history',
8
+ badgeCount: 0,
9
+ },
10
+ ],
11
+
12
+ switchs: [
13
+ {
14
+ label: '联网搜索',
15
+ name: 'netSearch',
16
+ type: 'checkbox',
17
+ defaultValue: false,
18
+ enabled: true,
19
+ icon: <Icon type='rag/url' size={12} />,
20
+ },
21
+ {
22
+ label: '推理思考',
23
+ name: 'reasoning',
24
+ type: 'checkbox',
25
+ defaultValue: false,
26
+ enabled: true,
27
+ icon: <Icon type='rag/think' size={12} />,
28
+ },
29
+ ],
30
+ };
31
+
32
+ export const MaxHistoryChatDisplay = 5000;
33
+ export const MaxHistoryItemDisplay = 3000;