cx-chat 0.0.1

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 (404) hide show
  1. package/.cursor/rules/i18n-cn-gloss-comments.mdc +31 -0
  2. package/.cursor/rules/list-page-view-pageconfig.mdc +32 -0
  3. package/.cursor/rules/no-over-defensive-programming.mdc +90 -0
  4. package/.cursor/rules/requirement-description-for-agent.mdc +33 -0
  5. package/.cursor/rules/use-showToast-not-antd-message.mdc +28 -0
  6. package/.docker/Dockerfile +7 -0
  7. package/.env +9 -0
  8. package/.env.development +7 -0
  9. package/.env.production +7 -0
  10. package/.gitlab-ci/docker-build.yaml +28 -0
  11. package/.gitlab-ci/k8s-deploy-dev-master.yaml +42 -0
  12. package/.gitlab-ci/npm-build.yaml +17 -0
  13. package/.gitlab-ci.yml +8 -0
  14. package/.k8s/0-namespace.yaml +6 -0
  15. package/.k8s/1-configmap-web.yaml +7 -0
  16. package/.k8s/1-nginx-conf-dev.yaml +110 -0
  17. package/.k8s/2-deployment.yaml +27 -0
  18. package/.k8s/3-service.yaml +16 -0
  19. package/.k8s/4-ingress-dev.yaml +30 -0
  20. package/.lingma/rules/use-showToast-not-antd-message.md +34 -0
  21. package/.nginx/nginx.conf +52 -0
  22. package/.prettierrc +9 -0
  23. package/README.md +1 -0
  24. package/eslint.config.js +32 -0
  25. package/index.html +13 -0
  26. package/package.json +67 -0
  27. package/postcss.config.js +6 -0
  28. package/public/favicon.ico +0 -0
  29. package/public/vite.svg +1 -0
  30. package/src/App.tsx +96 -0
  31. package/src/_doc/0.docs-overview.md +28 -0
  32. package/src/_doc/cx-ui/0.docs-overview.md +30 -0
  33. package/src/_doc/cx-ui/comp.1.cx-ui-overview.md +82 -0
  34. package/src/_doc/cx-ui/comp.2.cx-modal.md +82 -0
  35. package/src/_doc/cx-ui/comp.3.cx-button.md +89 -0
  36. package/src/_doc/cx-ui/comp.4.cx-form.md +72 -0
  37. package/src/_doc/cx-ui/comp.5.cx-fields.md +76 -0
  38. package/src/_doc/cx-ui/comp.6.cx-tag.md +57 -0
  39. package/src/_doc/cx-ui/comp.7.cx-empty-state.md +29 -0
  40. package/src/_doc/meta/0.docs-overview.md +24 -0
  41. package/src/_doc/meta/comp.1.enum-runtime.md +33 -0
  42. package/src/_doc/meta/comp.2.dict-runtime.md +39 -0
  43. package/src/_doc/router/0.docs-overview.md +14 -0
  44. package/src/_doc/router/guide.1.menu-component-config.md +181 -0
  45. package/src/_doc/router/guide.2.router-auto-registration.md +114 -0
  46. package/src/_doc/table-view/0.docs-overview.md +30 -0
  47. package/src/_doc/table-view/comp.1.table-view.md +542 -0
  48. package/src/_doc/table-view/props.1.create-table-view-config.md +193 -0
  49. package/src/_doc/table-view/props.2.table-view-search-fields.md +106 -0
  50. package/src/api/_mock/README.md +340 -0
  51. package/src/api/_mock/api.ts +1642 -0
  52. package/src/api/_mock/bundle-shim.ts +16 -0
  53. package/src/api/_mock/handler-shim.ts +6 -0
  54. package/src/api/_mock/handler.ts +458 -0
  55. package/src/api/_mock/index.ts +711 -0
  56. package/src/api/_mock/interceptor.ts +15 -0
  57. package/src/api/_mock/mod.ts +12 -0
  58. package/src/api/_mock/utils.ts +65 -0
  59. package/src/api/base/memory.js +24 -0
  60. package/src/api/chat.js +210 -0
  61. package/src/api/common/auth.js +70 -0
  62. package/src/api/menus/business-rules.js +76 -0
  63. package/src/api/menus/feedback.js +102 -0
  64. package/src/api/menus/knowledge.js +159 -0
  65. package/src/api/menus/model-metadata/manage.js +70 -0
  66. package/src/api/menus/model-metadata/role.js +50 -0
  67. package/src/api/menus/model-metadata/training-detail-mock-data.js +569 -0
  68. package/src/api/menus/model-metadata/training.js +28 -0
  69. package/src/api/menus/skill.js +40 -0
  70. package/src/api/system/agent-config.js +16 -0
  71. package/src/api/system/department.js +94 -0
  72. package/src/api/system/dict.js +86 -0
  73. package/src/api/system/menu.js +37 -0
  74. package/src/api/system/permission.js +26 -0
  75. package/src/api/system/role.js +34 -0
  76. package/src/api/system/sys-config.js +16 -0
  77. package/src/api/system/sys-log.js +17 -0
  78. package/src/api/system/user.js +75 -0
  79. package/src/api/upload.js +39 -0
  80. package/src/assets/react.svg +1 -0
  81. package/src/components/auth/current-user-avatar.tsx +77 -0
  82. package/src/components/common/code-view.tsx +149 -0
  83. package/src/components/common/detail-link.tsx +67 -0
  84. package/src/components/common/error-boundary.tsx +98 -0
  85. package/src/components/common/language-switcher.tsx +91 -0
  86. package/src/components/common/lite-table/index.tsx +135 -0
  87. package/src/components/common/md-editor.tsx +126 -0
  88. package/src/components/common/modal/confirm-dialog.tsx +113 -0
  89. package/src/components/common/modal/dep-user-select-multi.tsx +324 -0
  90. package/src/components/common/modal/dep-user-select.tsx +249 -0
  91. package/src/components/common/modal/user-select-multi.tsx +266 -0
  92. package/src/components/common/pagination.tsx +472 -0
  93. package/src/components/common/path.tsx +175 -0
  94. package/src/components/common/system-logo-mark.tsx +48 -0
  95. package/src/components/cx-ui/button/index.less +208 -0
  96. package/src/components/cx-ui/button/index.tsx +611 -0
  97. package/src/components/cx-ui/checkbox/index.tsx +78 -0
  98. package/src/components/cx-ui/date-picker/index.less +17 -0
  99. package/src/components/cx-ui/date-picker/index.tsx +193 -0
  100. package/src/components/cx-ui/drawer/index.tsx +47 -0
  101. package/src/components/cx-ui/empty-state/index.tsx +20 -0
  102. package/src/components/cx-ui/floating-shell/CxFloatingShell.tsx +89 -0
  103. package/src/components/cx-ui/floating-shell/cx-floating-shell.less +283 -0
  104. package/src/components/cx-ui/floating-shell/has-floating-value.ts +41 -0
  105. package/src/components/cx-ui/form/CxForm.tsx +15 -0
  106. package/src/components/cx-ui/form/index.tsx +20 -0
  107. package/src/components/cx-ui/form-item/index.less +26 -0
  108. package/src/components/cx-ui/form-item/index.tsx +36 -0
  109. package/src/components/cx-ui/index.ts +70 -0
  110. package/src/components/cx-ui/input/auto-complete.tsx +134 -0
  111. package/src/components/cx-ui/input/index.tsx +259 -0
  112. package/src/components/cx-ui/input-number/index.jsx +66 -0
  113. package/src/components/cx-ui/modal/index.jsx +212 -0
  114. package/src/components/cx-ui/modal/index.less +144 -0
  115. package/src/components/cx-ui/modal/useCxModal.ts +125 -0
  116. package/src/components/cx-ui/multi-select/index.jsx +74 -0
  117. package/src/components/cx-ui/multi-select/index.less +40 -0
  118. package/src/components/cx-ui/multi-select/index2.tsx +361 -0
  119. package/src/components/cx-ui/radio/index.tsx +33 -0
  120. package/src/components/cx-ui/range-picker/index.less +65 -0
  121. package/src/components/cx-ui/range-picker/index.tsx +219 -0
  122. package/src/components/cx-ui/select/index.less +34 -0
  123. package/src/components/cx-ui/select/index.tsx +196 -0
  124. package/src/components/cx-ui/skeleton/index.tsx +12 -0
  125. package/src/components/cx-ui/steps/index.tsx +14 -0
  126. package/src/components/cx-ui/styles/_tokens.less +79 -0
  127. package/src/components/cx-ui/styles/index.less +246 -0
  128. package/src/components/cx-ui/switch/index.less +106 -0
  129. package/src/components/cx-ui/switch/index.tsx +120 -0
  130. package/src/components/cx-ui/table/index.less +160 -0
  131. package/src/components/cx-ui/table/index.tsx +152 -0
  132. package/src/components/cx-ui/tabs/index.less +15 -0
  133. package/src/components/cx-ui/tabs/index.tsx +34 -0
  134. package/src/components/cx-ui/tag/index.less +51 -0
  135. package/src/components/cx-ui/tag/index.tsx +140 -0
  136. package/src/components/cx-ui/timeline/index.tsx +14 -0
  137. package/src/components/cx-ui/tooltip/index.tsx +67 -0
  138. package/src/components/cx-ui/tree/index.tsx +193 -0
  139. package/src/components/cx-ui/tree-select/index.jsx +91 -0
  140. package/src/components/cx-ui/tree-select/index.less +27 -0
  141. package/src/components/cx-ui/upload-file/index.less +223 -0
  142. package/src/components/cx-ui/upload-file/index.tsx +640 -0
  143. package/src/components/cx-ui/upload-img/index.tsx +291 -0
  144. package/src/components/layout/components/Header.tsx +216 -0
  145. package/src/components/layout/components/Sidebar.tsx +717 -0
  146. package/src/components/layout/index.tsx +95 -0
  147. package/src/components/table-view/components/search-area.tsx +411 -0
  148. package/src/components/table-view/components/table-view-config.tsx +528 -0
  149. package/src/components/table-view/components/table-view.types.ts +478 -0
  150. package/src/components/table-view/components/tree-api-normalize.ts +38 -0
  151. package/src/components/table-view/components/tree-data-annotate.ts +31 -0
  152. package/src/components/table-view/components/tree-sidebar.tsx +74 -0
  153. package/src/components/table-view/index.tsx +61 -0
  154. package/src/components/table-view/list-page-view.tsx +1049 -0
  155. package/src/components/table-view/select-table-view.tsx +1094 -0
  156. package/src/components/table-view/styles/select-table-view.less +51 -0
  157. package/src/config/default-system-name.ts +9 -0
  158. package/src/config/system.ts +165 -0
  159. package/src/constants/countryCodes.ts +3 -0
  160. package/src/contexts/AuthContext.tsx +256 -0
  161. package/src/contexts/ChatContext.tsx +839 -0
  162. package/src/contexts/MenuContext.tsx +62 -0
  163. package/src/contexts/ToastContext.tsx +181 -0
  164. package/src/hooks/useCopyToClipboard.ts +47 -0
  165. package/src/hooks/useModalSubmit.ts +104 -0
  166. package/src/hooks/useRouter.ts +240 -0
  167. package/src/hooks/useStepForm.ts +46 -0
  168. package/src/hooks/useStickyHeader.ts +42 -0
  169. package/src/hooks/useThreadActions.ts +105 -0
  170. package/src/hooks/useUserPreferences.ts +101 -0
  171. package/src/http/axios.js +372 -0
  172. package/src/http/mock.interceptor.ts +9 -0
  173. package/src/http/obfuscationKey.ts +41 -0
  174. package/src/i18n.ts +60 -0
  175. package/src/index.js +1 -0
  176. package/src/index.less +169 -0
  177. package/src/locales/en/auth.ts +70 -0
  178. package/src/locales/en/base/memory.ts +28 -0
  179. package/src/locales/en/base/settings.ts +41 -0
  180. package/src/locales/en/chat.ts +40 -0
  181. package/src/locales/en/common.ts +173 -0
  182. package/src/locales/en/enum.ts +27 -0
  183. package/src/locales/en/menus/business-rules.ts +48 -0
  184. package/src/locales/en/menus/feedback.ts +62 -0
  185. package/src/locales/en/menus/knowledge.ts +120 -0
  186. package/src/locales/en/menus/model-metadata/index.ts +10 -0
  187. package/src/locales/en/menus/model-metadata/manage.ts +151 -0
  188. package/src/locales/en/menus/model-metadata/role.ts +48 -0
  189. package/src/locales/en/menus/model-metadata/training.ts +65 -0
  190. package/src/locales/en/menus/skill.ts +34 -0
  191. package/src/locales/en/system/agent-config.ts +34 -0
  192. package/src/locales/en/system/department.ts +68 -0
  193. package/src/locales/en/system/dict.ts +44 -0
  194. package/src/locales/en/system/menu.ts +45 -0
  195. package/src/locales/en/system/permission.ts +89 -0
  196. package/src/locales/en/system/role.ts +25 -0
  197. package/src/locales/en/system/sys-config.ts +33 -0
  198. package/src/locales/en/system/sys-log.ts +38 -0
  199. package/src/locales/en/system/user.ts +113 -0
  200. package/src/locales/en.ts +68 -0
  201. package/src/locales/zh/auth.ts +70 -0
  202. package/src/locales/zh/base/memory.ts +29 -0
  203. package/src/locales/zh/base/settings.ts +41 -0
  204. package/src/locales/zh/chat.ts +47 -0
  205. package/src/locales/zh/common.ts +178 -0
  206. package/src/locales/zh/enum.ts +28 -0
  207. package/src/locales/zh/menus/business-rules.ts +47 -0
  208. package/src/locales/zh/menus/feedback.ts +62 -0
  209. package/src/locales/zh/menus/knowledge.ts +117 -0
  210. package/src/locales/zh/menus/model-metadata/index.ts +10 -0
  211. package/src/locales/zh/menus/model-metadata/manage.ts +151 -0
  212. package/src/locales/zh/menus/model-metadata/role.ts +47 -0
  213. package/src/locales/zh/menus/model-metadata/training.ts +64 -0
  214. package/src/locales/zh/menus/skill.ts +34 -0
  215. package/src/locales/zh/system/agent-config.ts +33 -0
  216. package/src/locales/zh/system/department.ts +69 -0
  217. package/src/locales/zh/system/dict.ts +44 -0
  218. package/src/locales/zh/system/menu.ts +47 -0
  219. package/src/locales/zh/system/permission.ts +94 -0
  220. package/src/locales/zh/system/role.ts +25 -0
  221. package/src/locales/zh/system/sys-config.ts +32 -0
  222. package/src/locales/zh/system/sys-log.ts +38 -0
  223. package/src/locales/zh/system/user.ts +114 -0
  224. package/src/locales/zh.ts +67 -0
  225. package/src/main.tsx +50 -0
  226. package/src/meta/const/index.ts +40 -0
  227. package/src/meta/index-dict.ts +56 -0
  228. package/src/meta/index-enum.ts +95 -0
  229. package/src/meta/index.ts +14 -0
  230. package/src/meta/module/dict-data/runtime.ts +199 -0
  231. package/src/meta/module/dict-data/types.ts +17 -0
  232. package/src/meta/module/enum-data/runtime.ts +75 -0
  233. package/src/meta/module/enum-data/types.ts +18 -0
  234. package/src/router/index.tsx +312 -0
  235. package/src/styles/AntdThemeProvider.tsx +40 -0
  236. package/src/styles/antd-theme.ts +20 -0
  237. package/src/styles/global.less +107 -0
  238. package/src/styles/variable.less +103 -0
  239. package/src/types/feedback.ts +43 -0
  240. package/src/types/index.ts +85 -0
  241. package/src/types/menu.ts +43 -0
  242. package/src/utils/aesUtil.ts +123 -0
  243. package/src/utils/chatUtils.ts +524 -0
  244. package/src/utils/cn.ts +6 -0
  245. package/src/utils/crypto.ts +164 -0
  246. package/src/utils/date.ts +72 -0
  247. package/src/utils/file-icons.tsx +79 -0
  248. package/src/utils/index.ts +168 -0
  249. package/src/utils/markdown-math-plugins.ts +21 -0
  250. package/src/utils/menuI18n.ts +305 -0
  251. package/src/utils/menuRouteRegistry.ts +78 -0
  252. package/src/utils/permission-crud.ts +147 -0
  253. package/src/utils/routeConfig.ts +350 -0
  254. package/src/utils/storage.ts +135 -0
  255. package/src/utils/toastBridge.ts +26 -0
  256. package/src/utils/url.ts +38 -0
  257. package/src/utils/validation.ts +16 -0
  258. package/src/views/auth/auth-code/index.less +169 -0
  259. package/src/views/auth/auth-code/index.module.less +174 -0
  260. package/src/views/auth/auth-code/index.tsx +233 -0
  261. package/src/views/auth/login.tsx +498 -0
  262. package/src/views/auth/register.tsx +388 -0
  263. package/src/views/base/memory/index.tsx +136 -0
  264. package/src/views/base/memory/modal/detail-modal.tsx +89 -0
  265. package/src/views/base/memory/modal/submit-modal.tsx +134 -0
  266. package/src/views/base/settings/index.tsx +657 -0
  267. package/src/views/chat/chat.less +323 -0
  268. package/src/views/chat/components/chat-input.tsx +298 -0
  269. package/src/views/chat/components/header-thread-title.tsx +210 -0
  270. package/src/views/chat/components/message-list/content-answer.tsx +100 -0
  271. package/src/views/chat/components/message-list/content-question.tsx +18 -0
  272. package/src/views/chat/components/message-list/index.tsx +520 -0
  273. package/src/views/chat/components/message-list/message-item.tsx +199 -0
  274. package/src/views/chat/components/message-list/preparation-demo-items.ts +147 -0
  275. package/src/views/chat/components/message-list/preparation-steps.tsx +506 -0
  276. package/src/views/chat/components/message-list/suggestion-list.tsx +36 -0
  277. package/src/views/chat/components/message-list/thinking-process.tsx +49 -0
  278. package/src/views/chat/components/message-list/toolbar.tsx +224 -0
  279. package/src/views/chat/components/message-list/use-message-list-scroll.ts +214 -0
  280. package/src/views/chat/components/references-knowledge/context.tsx +57 -0
  281. package/src/views/chat/components/references-knowledge/index.ts +9 -0
  282. package/src/views/chat/components/references-knowledge/modal/knowledge-detail-drawer.tsx +556 -0
  283. package/src/views/chat/components/references-knowledge/modal/knowledge-doc-detail-drawer.tsx +529 -0
  284. package/src/views/chat/components/references-knowledge/panel.tsx +115 -0
  285. package/src/views/chat/hooks/use-chat-common.ts +19 -0
  286. package/src/views/chat/index-session.tsx +647 -0
  287. package/src/views/chat/index.tsx +127 -0
  288. package/src/views/page-error/401.tsx +56 -0
  289. package/src/views/page-error/404.tsx +56 -0
  290. package/src/views/page-menus/business-rules/index.tsx +376 -0
  291. package/src/views/page-menus/business-rules/modal/detail-modal.tsx +186 -0
  292. package/src/views/page-menus/business-rules/modal/scope-modal.tsx +272 -0
  293. package/src/views/page-menus/business-rules/modal/submit-modal.tsx +142 -0
  294. package/src/views/page-menus/feedback/components/feedback-dataset-list.tsx +471 -0
  295. package/src/views/page-menus/feedback/index.tsx +166 -0
  296. package/src/views/page-menus/feedback/modal/export-feedback-modal.tsx +367 -0
  297. package/src/views/page-menus/knowledge/components/doc-editor-by-type.tsx +32 -0
  298. package/src/views/page-menus/knowledge/components/doc-editor-type-file.tsx +330 -0
  299. package/src/views/page-menus/knowledge/detail.tsx +600 -0
  300. package/src/views/page-menus/knowledge/index.tsx +337 -0
  301. package/src/views/page-menus/knowledge/modal/detail-modal.tsx +618 -0
  302. package/src/views/page-menus/knowledge/modal/doc-detail-modal.tsx +550 -0
  303. package/src/views/page-menus/knowledge/modal/doc-parse.ts +99 -0
  304. package/src/views/page-menus/knowledge/modal/doc-submit-modal.tsx +349 -0
  305. package/src/views/page-menus/knowledge/modal/doc-type-picker-modal.tsx +88 -0
  306. package/src/views/page-menus/knowledge/modal/knowledge-user-select-modal.tsx +283 -0
  307. package/src/views/page-menus/knowledge/modal/submit-modal.tsx +179 -0
  308. package/src/views/page-menus/model-metadata/manage/components/metadata-detail-schema-tab.tsx +114 -0
  309. package/src/views/page-menus/model-metadata/manage/components/step1-basic-info.tsx +232 -0
  310. package/src/views/page-menus/model-metadata/manage/components/step2-schema.tsx +316 -0
  311. package/src/views/page-menus/model-metadata/manage/components/step3-permissions.tsx +134 -0
  312. package/src/views/page-menus/model-metadata/manage/components/step4-documents.tsx +134 -0
  313. package/src/views/page-menus/model-metadata/manage/components/step5-example-sql.tsx +101 -0
  314. package/src/views/page-menus/model-metadata/manage/components/submit-add.tsx +338 -0
  315. package/src/views/page-menus/model-metadata/manage/components/submit-edit.tsx +276 -0
  316. package/src/views/page-menus/model-metadata/manage/detail.tsx +298 -0
  317. package/src/views/page-menus/model-metadata/manage/hooks/model-metadata-submit-shared.ts +113 -0
  318. package/src/views/page-menus/model-metadata/manage/hooks/use-model-metadata-item-state.ts +20 -0
  319. package/src/views/page-menus/model-metadata/manage/index.tsx +304 -0
  320. package/src/views/page-menus/model-metadata/manage/modal/components/table-schema.ts +374 -0
  321. package/src/views/page-menus/model-metadata/manage/modal/components/use-table-detail-tabs.tsx +151 -0
  322. package/src/views/page-menus/model-metadata/manage/modal/components/use-table-submit-tabs.tsx +423 -0
  323. package/src/views/page-menus/model-metadata/manage/modal/detail-modal.tsx +218 -0
  324. package/src/views/page-menus/model-metadata/manage/modal/submit-modal.tsx +261 -0
  325. package/src/views/page-menus/model-metadata/manage/modal/table-detail-modal.tsx +196 -0
  326. package/src/views/page-menus/model-metadata/manage/modal/table-submit-modal.tsx +229 -0
  327. package/src/views/page-menus/model-metadata/manage/submit.tsx +31 -0
  328. package/src/views/page-menus/model-metadata/role/index.tsx +207 -0
  329. package/src/views/page-menus/model-metadata/role/modal/detail-modal.tsx +97 -0
  330. package/src/views/page-menus/model-metadata/role/modal/role-assign-users-modal.tsx +254 -0
  331. package/src/views/page-menus/model-metadata/role/modal/role-assign-users-panel.tsx +393 -0
  332. package/src/views/page-menus/model-metadata/role/modal/role-assign-users-utils.ts +120 -0
  333. package/src/views/page-menus/model-metadata/role/modal/role-permission-assign-panel.tsx +698 -0
  334. package/src/views/page-menus/model-metadata/role/modal/role-permission-modal.tsx +237 -0
  335. package/src/views/page-menus/model-metadata/role/modal/submit-modal.tsx +135 -0
  336. package/src/views/page-menus/model-metadata/training/components/detail-records/index.ts +4 -0
  337. package/src/views/page-menus/model-metadata/training/components/detail-records/node-card.tsx +72 -0
  338. package/src/views/page-menus/model-metadata/training/components/detail-records/summary-lines.ts +196 -0
  339. package/src/views/page-menus/model-metadata/training/components/detail-records/summary-list.tsx +153 -0
  340. package/src/views/page-menus/model-metadata/training/components/detail-records/timeline.tsx +103 -0
  341. package/src/views/page-menus/model-metadata/training/components/detail-records/types.ts +82 -0
  342. package/src/views/page-menus/model-metadata/training/detail.tsx +159 -0
  343. package/src/views/page-menus/model-metadata/training/index.tsx +236 -0
  344. package/src/views/page-menus/model-metadata/training/modal/update-detail-modal.tsx +154 -0
  345. package/src/views/page-menus/skill/index.tsx +201 -0
  346. package/src/views/page-menus/skill/modal/detail-modal.tsx +156 -0
  347. package/src/views/page-menus/skill/modal/submit-modal.tsx +214 -0
  348. package/src/views/page-system/agent-config/index.tsx +370 -0
  349. package/src/views/page-system/department/departmentFormShared.ts +36 -0
  350. package/src/views/page-system/department/index.tsx +541 -0
  351. package/src/views/page-system/department/modal/detail-modal.tsx +94 -0
  352. package/src/views/page-system/department/modal/member-role-modal.tsx +128 -0
  353. package/src/views/page-system/department/modal/submit-modal.tsx +265 -0
  354. package/src/views/page-system/dict/index.tsx +440 -0
  355. package/src/views/page-system/dict/modal/cate-submit-modal.tsx +315 -0
  356. package/src/views/page-system/dict/modal/submit-modal.tsx +184 -0
  357. package/src/views/page-system/logs/components/index.ts +3 -0
  358. package/src/views/page-system/logs/components/log-message-demo.tsx +30 -0
  359. package/src/views/page-system/logs/components/log-message-stream.ts +184 -0
  360. package/src/views/page-system/logs/components/message-list/content-answer.tsx +100 -0
  361. package/src/views/page-system/logs/components/message-list/content-question.tsx +18 -0
  362. package/src/views/page-system/logs/components/message-list/index.tsx +515 -0
  363. package/src/views/page-system/logs/components/message-list/message-item.tsx +193 -0
  364. package/src/views/page-system/logs/components/message-list/preparation-demo-items.ts +147 -0
  365. package/src/views/page-system/logs/components/message-list/preparation-steps.tsx +506 -0
  366. package/src/views/page-system/logs/components/message-list/suggestion-list.tsx +36 -0
  367. package/src/views/page-system/logs/components/message-list/thinking-process.tsx +49 -0
  368. package/src/views/page-system/logs/components/message-list/toolbar.tsx +134 -0
  369. package/src/views/page-system/logs/components/message-list/use-message-list-scroll.ts +214 -0
  370. package/src/views/page-system/logs/components/message-modal.tsx +239 -0
  371. package/src/views/page-system/logs/index.tsx +132 -0
  372. package/src/views/page-system/logs/modal/detail-modal.tsx +157 -0
  373. package/src/views/page-system/menu/components/menuFormShared.ts +283 -0
  374. package/src/views/page-system/menu/index.less +12 -0
  375. package/src/views/page-system/menu/index.tsx +410 -0
  376. package/src/views/page-system/menu/modal/icon-modal.less +51 -0
  377. package/src/views/page-system/menu/modal/icon-modal.tsx +87 -0
  378. package/src/views/page-system/menu/modal/submit-modal.tsx +263 -0
  379. package/src/views/page-system/permission/index.tsx +562 -0
  380. package/src/views/page-system/permission/modal/detail-modal.tsx +179 -0
  381. package/src/views/page-system/permission/modal/submit-modal.less +146 -0
  382. package/src/views/page-system/permission/modal/submit-modal.tsx +650 -0
  383. package/src/views/page-system/role/index.tsx +163 -0
  384. package/src/views/page-system/role/modal/detail-modal.tsx +127 -0
  385. package/src/views/page-system/role/modal/permission-assign-group-rules.ts +86 -0
  386. package/src/views/page-system/role/modal/permission-modal.tsx +111 -0
  387. package/src/views/page-system/role/modal/role-modal-shell-styles.ts +21 -0
  388. package/src/views/page-system/role/modal/role-permission-assign-panel.tsx +916 -0
  389. package/src/views/page-system/role/modal/role-permission-assign-shared.ts +1047 -0
  390. package/src/views/page-system/role/modal/submit-modal.tsx +193 -0
  391. package/src/views/page-system/sys-config/index.tsx +294 -0
  392. package/src/views/page-system/user/components/user-role-column.tsx +87 -0
  393. package/src/views/page-system/user/index.tsx +439 -0
  394. package/src/views/page-system/user/modal/assign-roles-modal.tsx +389 -0
  395. package/src/views/page-system/user/modal/detail-modal.tsx +72 -0
  396. package/src/views/page-system/user/modal/modal-style/submit-modal.less +40 -0
  397. package/src/views/page-system/user/modal/submit-modal.less +40 -0
  398. package/src/views/page-system/user/modal/submit-modal.tsx +287 -0
  399. package/src/views/page-system/user/userFormShared.ts +51 -0
  400. package/tailwind.config.js +17 -0
  401. package/tsconfig.app.json +48 -0
  402. package/tsconfig.json +11 -0
  403. package/tsconfig.node.json +26 -0
  404. package/vite.config.ts +264 -0
@@ -0,0 +1,647 @@
1
+ import './chat.less'
2
+
3
+ import { useQuery, useQueryClient } from '@tanstack/react-query'
4
+ import { Loader2 } from 'lucide-react'
5
+ import React, { useCallback, useEffect, useMemo, useRef, useState } from 'react'
6
+ import { useTranslation } from 'react-i18next'
7
+ import {
8
+ Navigate,
9
+ useLocation,
10
+ useOutletContext,
11
+ useParams,
12
+ } from 'react-router-dom'
13
+
14
+ import {
15
+ deleteChatMessageAPI,
16
+ feedbackChatMessageAPI,
17
+ getChatPromptAPI,
18
+ getChatSessionsAPI,
19
+ postChatMessagesAPI,
20
+ } from '@/api/chat'
21
+ import { getKnowledgeNameListAPI } from '@/api/menus/knowledge'
22
+ import { useAuth } from '@/contexts/AuthContext'
23
+ import { useChatContext } from '@/contexts/ChatContext'
24
+ import { useToast } from '@/contexts/ToastContext'
25
+ import type { MainLayoutContext } from '@/hooks/useStickyHeader'
26
+ import type { Message } from '@/types'
27
+ import {
28
+ applySessionPromptBinding,
29
+ buildChatFeedbackEvaluationType,
30
+ evaluationTypeAfterFeedbackRequest,
31
+ findLastAssistantMessage,
32
+ findLastNonSystemMessage,
33
+ getMessageEvaluationDisplay,
34
+ mergeStreamingPlaceholderThread,
35
+ normalizeMessagesList,
36
+ normalizeSessionsToThreads,
37
+ parseChatSessionPromptSuggestions,
38
+ resolveMessageTurnIdForFeedback,
39
+ } from '@/utils/chatUtils'
40
+ import {
41
+ getChatEnableDataAnalysis,
42
+ getChatEnableKnowledgeBase,
43
+ getChatEnableNetwork,
44
+ } from '@/utils/storage'
45
+
46
+ import { useDeletingChatSessionId } from '@/hooks/useThreadActions'
47
+ import ChatInput from './components/chat-input'
48
+ import HeaderThreadTitle from './components/header-thread-title'
49
+ import MessageList from './components/message-list'
50
+ import PreparationSteps, {
51
+ CHAT_PREPARATION_DEMO_ITEMS,
52
+ isChatPreparationDemoEnabled,
53
+ } from './components/message-list/preparation-steps'
54
+ import {
55
+ ReferencesKnowledgeProvider,
56
+ useReferencesKnowledge,
57
+ } from './components/references-knowledge'
58
+ import { useChatCommon } from './hooks/use-chat-common'
59
+
60
+ /**
61
+ * 开发态样式联调:给历史消息注入「参考资料」demo 数据,便于在后台未接入时预览抽屉展示。
62
+ */
63
+ function injectReferencesKnowledgeDemo(messages: Message[]): Message[] {
64
+ // const demoItems: NonNullable<Message['references_knowledge_items']> = [
65
+ // {
66
+ // kb_code: 'kb_000002',
67
+ // doc_code: 'kb_000002_20260415000003',
68
+ // page_content: '王刚平时在公司食堂做饭,擅长江湖菜',
69
+ // },
70
+ // {
71
+ // kb_code: 'kb_000002',
72
+ // doc_code: 'kb_000002_20260415000002',
73
+ // page_content: '小林的平时工作是负责运维公司服务器',
74
+ // },
75
+ // {
76
+ // kb_code: 'kb_000002',
77
+ // doc_code: 'kb_000002_20260415000004',
78
+ // page_content: '小鹏在公司天天摸鱼啥也不干',
79
+ // },
80
+ // ]
81
+
82
+ for (let i = messages.length - 1; i >= 0; i -= 1) {
83
+ const m = messages[i]
84
+ if (m.role !== 'assistant') continue
85
+ if (!m.content) continue
86
+ if (m.is_streaming) continue
87
+ if (m.references_knowledge_items && m.references_knowledge_items.length > 0)
88
+ return messages
89
+ const next = [...messages]
90
+ // references_knowledge_items: demoItems
91
+ next[i] = { ...m, }
92
+ return next
93
+ }
94
+ return messages
95
+ }
96
+
97
+ /**
98
+ * 解析知识库名称接口,生成 code -> name 的映射表。
99
+ */
100
+ function buildKnowledgeCodeNameMap(raw: unknown): Record<string, string> {
101
+ const root =
102
+ raw && typeof raw === 'object' && 'data' in raw
103
+ ? (raw as { data?: unknown }).data
104
+ : raw
105
+ const list = Array.isArray(root)
106
+ ? root
107
+ : root && typeof root === 'object' && Array.isArray((root as { data?: unknown }).data)
108
+ ? ((root as { data?: unknown }).data as unknown[])
109
+ : []
110
+ const map: Record<string, string> = {}
111
+ for (const item of list) {
112
+ if (!item || typeof item !== 'object') continue
113
+ const row = item as Record<string, unknown>
114
+ const code = typeof row.code === 'string' ? row.code.trim() : ''
115
+ const name = typeof row.name === 'string' ? row.name.trim() : ''
116
+ if (!code || !name) continue
117
+ map[code] = name
118
+ }
119
+ return map
120
+ }
121
+
122
+ /**
123
+ * 给参考资料项补充知识库名称,供界面显示“参考资料(名称)/知识库:名称”。
124
+ */
125
+ function attachKnowledgeNameToReferences(
126
+ refs: NonNullable<Message['references_knowledge_items']> | undefined,
127
+ knowledgeCodeNameMap: Record<string, string>,
128
+ ): NonNullable<Message['references_knowledge_items']> | undefined {
129
+ if (!refs || refs.length === 0) return refs
130
+ return refs.map((item) => {
131
+ /** 通过知识库 code 映射知识库名称 */
132
+ const kbCode = typeof item.kb_code === 'string' ? item.kb_code.trim() : ''
133
+ /** 通过文档 code 映射文档名称 */
134
+ const docCode = typeof item.doc_code === 'string' ? item.doc_code.trim() : ''
135
+ const kbName = kbCode ? knowledgeCodeNameMap[kbCode] : ''
136
+ const docName = docCode ? knowledgeCodeNameMap[docCode] : ''
137
+ if (!kbName && !docName) return item
138
+ return {
139
+ ...item,
140
+ ...(kbName ? { kb_name: kbName } : {}),
141
+ ...(docName ? { doc_name: docName } : {}),
142
+ }
143
+ })
144
+ }
145
+
146
+ /**
147
+ * 参考资料感知输入框:根据参考资料面板状态自动调整右侧内边距。
148
+ */
149
+ function ReferencesKnowledgeAwareInput({
150
+ children,
151
+ }: {
152
+ children: React.ReactNode
153
+ }) {
154
+ const refs = useReferencesKnowledge()
155
+ return (
156
+ <div className={`w-full ${refs.isOpen ? 'pr-[420px]' : ''}`}>
157
+ {children}
158
+ </div>
159
+ )
160
+ }
161
+
162
+ const ChatSessionView: React.FC<{ sessionId: string }> = ({ sessionId }) => {
163
+ const location = useLocation()
164
+ const chatPrepDemoEnabled =
165
+ import.meta.env.DEV && isChatPreparationDemoEnabled(location.search)
166
+ /** 当前正在删除的会话 id,用于判断本页是否显示删除遮罩 */
167
+ const deletingSessionId = useDeletingChatSessionId()
168
+ /** 路由参数解码后的会话 id,供接口请求和对比使用 */
169
+ const decodedSessionId = decodeURIComponent(sessionId)
170
+ /** 是否正在删除当前会话 */
171
+ const isDeletingThisSession =
172
+ deletingSessionId != null && deletingSessionId === decodedSessionId
173
+
174
+ /** React Query 客户端,用于读写消息缓存 */
175
+ const queryClient = useQueryClient()
176
+ /** 全局消息提示方法 */
177
+ const { showToast } = useToast()
178
+ /** 国际化函数 */
179
+ const { t } = useTranslation()
180
+ /** 权限判断方法 */
181
+ const { hasPermission } = useAuth()
182
+ /** 布局头部标题设置方法 */
183
+ const { setHeaderTitle } = useOutletContext<MainLayoutContext>()
184
+ const {
185
+ activeThreadId,
186
+ isStreaming,
187
+ streamingContent,
188
+ streamingPlaceholderThread,
189
+ sendMessage,
190
+ regenerateMessage,
191
+ } = useChatContext()
192
+ /** 会话输入区通用能力(创建权限、停止生成) */
193
+ const { canCreate, handleStop } = useChatCommon()
194
+ /** 消息选择模式,开启后隐藏输入框并锁定标题编辑 */
195
+ const [messageSelectMode, setMessageSelectMode] = useState(false)
196
+ /** 向上翻页加载的更早消息(与 useQuery 最新一页拼接) */
197
+ const [olderMessages, setOlderMessages] = useState<Message[]>([])
198
+ const [messagesHasMore, setMessagesHasMore] = useState(true)
199
+ const [isLoadingMoreMessages, setIsLoadingMoreMessages] = useState(false)
200
+
201
+ /** 是否有“重新生成”权限 */
202
+ const canResend = hasPermission('system:chat:re-send')
203
+ /** 是否有“删除消息”权限 */
204
+ const canDeleteMessage = hasPermission('system:chat:message-delete')
205
+
206
+ const { data: threadsRemote = [] } = useQuery({
207
+ queryKey: ['threads'],
208
+ queryFn: async () => {
209
+ const raw = await getChatSessionsAPI()
210
+ return normalizeSessionsToThreads(raw)
211
+ },
212
+ enabled: hasPermission('system:chat:sessions'),
213
+ })
214
+
215
+ const threads = useMemo(
216
+ () =>
217
+ mergeStreamingPlaceholderThread(threadsRemote, {
218
+ placeholder: streamingPlaceholderThread,
219
+ activeThreadId,
220
+ isStreaming,
221
+ }),
222
+ [threadsRemote, streamingPlaceholderThread, activeThreadId, isStreaming],
223
+ )
224
+
225
+ /** 根据当前会话和选择状态设置页面头部标题 */
226
+ useEffect(() => {
227
+ setHeaderTitle(
228
+ <HeaderThreadTitle
229
+ threadId={sessionId}
230
+ allThreads={threads}
231
+ titleReadOnly={messageSelectMode}
232
+ />,
233
+ )
234
+ return () => setHeaderTitle(null)
235
+ }, [sessionId, setHeaderTitle, threads, messageSelectMode])
236
+
237
+ useEffect(() => {
238
+ setOlderMessages([])
239
+ setMessagesHasMore(true)
240
+ }, [sessionId])
241
+
242
+ const {
243
+ data: messagesLatest = [],
244
+ isLoading: isMessagesLoading,
245
+ error: messagesError,
246
+ } = useQuery({
247
+ queryKey: ['messages', sessionId],
248
+ queryFn: async ({ queryKey }) => {
249
+ const sid = String(queryKey[1] ?? '')
250
+ const raw = await postChatMessagesAPI({ sessionId: sid, round: 5 })
251
+ const list = normalizeMessagesList(raw)
252
+ setOlderMessages([])
253
+ setMessagesHasMore(list.length > 0)
254
+ return list
255
+ },
256
+ enabled: !!sessionId && hasPermission('system:chat:message-list'),
257
+ // 避免与 ChatContext 乐观更新 / 流式结束写缓存打架;仍可在需要处 invalidate
258
+ staleTime: 30_000,
259
+ refetchOnWindowFocus: false,
260
+ })
261
+
262
+ const messages = useMemo(() => {
263
+ if (olderMessages.length === 0) return messagesLatest
264
+ const latestIds = new Set(messagesLatest.map((m) => m.id))
265
+ const older = olderMessages.filter((m) => !latestIds.has(m.id))
266
+ return [...older, ...messagesLatest]
267
+ }, [olderMessages, messagesLatest])
268
+
269
+ const handleLoadMoreMessages = useCallback(async () => {
270
+ if (isLoadingMoreMessages || !messagesHasMore || isMessagesLoading) return
271
+ // 游标:上一次请求返回的第一条(后台顺序已正确,往前拼在列表头部)
272
+ const cursorMsg =
273
+ olderMessages.length > 0 ? olderMessages[0] : messagesLatest[0]
274
+ if (!cursorMsg) return
275
+ const turnId = resolveMessageTurnIdForFeedback(cursorMsg)
276
+ if (!turnId) {
277
+ setMessagesHasMore(false)
278
+ return
279
+ }
280
+ setIsLoadingMoreMessages(true)
281
+ try {
282
+ const raw = await postChatMessagesAPI({
283
+ sessionId: decodedSessionId,
284
+ round: 5,
285
+ turnId,
286
+ })
287
+ const batch = normalizeMessagesList(raw)
288
+ if (batch.length === 0) {
289
+ setMessagesHasMore(false)
290
+ } else {
291
+ setOlderMessages((prev) => {
292
+ const ids = new Set([...prev, ...messagesLatest].map((m) => m.id))
293
+ const merged = batch.filter((m) => !ids.has(m.id))
294
+ return [...merged, ...prev]
295
+ })
296
+ }
297
+ } catch (e) {
298
+ console.error('[chat-session] load more messages failed', e)
299
+ } finally {
300
+ setIsLoadingMoreMessages(false)
301
+ }
302
+ }, [
303
+ isLoadingMoreMessages,
304
+ messagesHasMore,
305
+ isMessagesLoading,
306
+ olderMessages,
307
+ decodedSessionId,
308
+ messagesLatest,
309
+ ])
310
+
311
+ /** 获取知识库 code-name 列表,用于将参考资料中的 kb_code 转成可读名称 */
312
+ const { data: knowledgeCodeNameMap = {} } = useQuery({
313
+ queryKey: ['knowledgeNameList'],
314
+ queryFn: async () => {
315
+ const raw = await getKnowledgeNameListAPI()
316
+ return buildKnowledgeCodeNameMap(raw)
317
+ },
318
+ })
319
+
320
+ /** 会话建议:仅在进入该会话(queryKey 随 sessionId 变化)时请求,不在发送/重试后刷新,避免回答结束后重绑建议导致界面跳动 */
321
+ const {
322
+ data: sessionPromptSuggestions = [],
323
+ dataUpdatedAt: sessionPromptDataUpdatedAt = 0,
324
+ } = useQuery({
325
+ queryKey: ['chatSessionPrompt', sessionId],
326
+ queryFn: async () => {
327
+ try {
328
+ const raw = await getChatPromptAPI({ sessionId: decodedSessionId })
329
+ return parseChatSessionPromptSuggestions(raw)
330
+ } catch {
331
+ return []
332
+ }
333
+ },
334
+ enabled: !!sessionId && hasPermission('system:chat:session-prompt'),
335
+ staleTime: 60_000,
336
+ refetchOnWindowFocus: false,
337
+ })
338
+
339
+ /** 接口用 sessionId 查;展示绑在「当次请求完成时」的最后一条非流式助手消息 id 上 */
340
+ const [sessionPromptBind, setSessionPromptBind] = useState<{
341
+ messageId: string
342
+ items: string[]
343
+ } | null>(null)
344
+ /** 记录最近一次会话建议请求完成时间,用于判断是否是新一轮建议 */
345
+ const lastSessionPromptFetchAtRef = useRef(0)
346
+
347
+ /** 会话切换时重置建议绑定状态 */
348
+ useEffect(() => {
349
+ lastSessionPromptFetchAtRef.current = 0
350
+ setSessionPromptBind(null)
351
+ }, [sessionId])
352
+
353
+ /** 拼接尾部消息:当前会话流式中时将流式内容拼到消息尾部 */
354
+ const tailMessages = useMemo(() => {
355
+ if (activeThreadId === sessionId && streamingContent) {
356
+ return [...messages, streamingContent]
357
+ }
358
+ return messages
359
+ }, [messages, streamingContent, activeThreadId, sessionId])
360
+
361
+ /** 根据尾部消息状态和建议更新时间,决定建议挂载到哪条助手消息 */
362
+ useEffect(() => {
363
+ const tail = findLastNonSystemMessage(tailMessages)
364
+ const lastAsst = findLastAssistantMessage(tailMessages)
365
+
366
+ if (tail?.role === 'user') {
367
+ setSessionPromptBind(null)
368
+ lastSessionPromptFetchAtRef.current = 0
369
+ return
370
+ }
371
+
372
+ if (!sessionPromptSuggestions.length) {
373
+ setSessionPromptBind(null)
374
+ lastSessionPromptFetchAtRef.current = 0
375
+ return
376
+ }
377
+
378
+ if (!lastAsst || lastAsst.is_streaming) {
379
+ setSessionPromptBind((prev) => {
380
+ if (!prev) return null
381
+ if (lastAsst?.is_streaming && prev.messageId === lastAsst.id) {
382
+ return null
383
+ }
384
+ return prev
385
+ })
386
+ return
387
+ }
388
+
389
+ const promptJustFetched =
390
+ sessionPromptDataUpdatedAt !== lastSessionPromptFetchAtRef.current
391
+ if (promptJustFetched) {
392
+ lastSessionPromptFetchAtRef.current = sessionPromptDataUpdatedAt
393
+ setSessionPromptBind({
394
+ messageId: lastAsst.id,
395
+ items: [...sessionPromptSuggestions],
396
+ })
397
+ return
398
+ }
399
+
400
+ setSessionPromptBind((prev) => {
401
+ if (!prev) return null
402
+ if (
403
+ tail?.role === 'assistant' &&
404
+ tail.id === prev.messageId &&
405
+ lastAsst.id === prev.messageId
406
+ ) {
407
+ return prev
408
+ }
409
+ return null
410
+ })
411
+ }, [
412
+ tailMessages,
413
+ sessionPromptSuggestions,
414
+ sessionPromptDataUpdatedAt,
415
+ ])
416
+
417
+ const messagesWithSessionPromptBind = useMemo(
418
+ () => applySessionPromptBinding(messages, sessionPromptBind),
419
+ [messages, sessionPromptBind],
420
+ )
421
+
422
+ /** 开发环境下给消息注入参考资料演示数据,便于联调展示 */
423
+ const messagesWithDemoRefs = useMemo(() => {
424
+ if (!import.meta.env.DEV) return messagesWithSessionPromptBind
425
+ return injectReferencesKnowledgeDemo(messagesWithSessionPromptBind)
426
+ }, [messagesWithSessionPromptBind])
427
+
428
+ /** 统一给参考资料补充知识库名称,避免界面直接展示知识库 code */
429
+ const messagesWithKnowledgeName = useMemo(
430
+ () =>
431
+ messagesWithDemoRefs.map((message) => ({
432
+ ...message,
433
+ references_knowledge_items: attachKnowledgeNameToReferences(
434
+ message.references_knowledge_items,
435
+ knowledgeCodeNameMap,
436
+ ),
437
+ })),
438
+ [messagesWithDemoRefs, knowledgeCodeNameMap],
439
+ )
440
+
441
+ /** 监听消息列表查询异常并打印日志 */
442
+ useEffect(() => {
443
+ if (messagesError) {
444
+ console.error('[chat-session] messages query error:', messagesError)
445
+ }
446
+ }, [messagesError])
447
+
448
+ /** 组装最终展示消息:当前会话流式中时追加流式内容 */
449
+ const displayMessages = useMemo(() => {
450
+ if (activeThreadId === sessionId && streamingContent) {
451
+ return [
452
+ ...messagesWithKnowledgeName,
453
+ {
454
+ ...streamingContent,
455
+ references_knowledge_items: attachKnowledgeNameToReferences(
456
+ streamingContent.references_knowledge_items,
457
+ knowledgeCodeNameMap,
458
+ ),
459
+ },
460
+ ]
461
+ }
462
+ return messagesWithKnowledgeName
463
+ }, [
464
+ messagesWithKnowledgeName,
465
+ streamingContent,
466
+ activeThreadId,
467
+ sessionId,
468
+ knowledgeCodeNameMap,
469
+ ])
470
+
471
+ /** 发送消息:处理空文本与联网参数后调用会话发送能力 */
472
+ const handleSend = useCallback(
473
+ async (
474
+ message: string,
475
+ opts?: {
476
+ enableNetwork?: boolean
477
+ enableKnowledgeBase?: boolean
478
+ enableDataAnalysis?: boolean
479
+ },
480
+ ) => {
481
+ const text = message.trim()
482
+ if (!text) return
483
+ const enableNetwork = opts?.enableNetwork ?? getChatEnableNetwork()
484
+ const enableKnowledgeBase =
485
+ opts?.enableKnowledgeBase ?? getChatEnableKnowledgeBase()
486
+ const enableDataAnalysis =
487
+ opts?.enableDataAnalysis ?? getChatEnableDataAnalysis()
488
+ await sendMessage(text, sessionId, {
489
+ enableNetwork,
490
+ enableKnowledgeBase,
491
+ enableDataAnalysis,
492
+ })
493
+ },
494
+ [sendMessage, sessionId],
495
+ )
496
+
497
+ /** 重新生成消息:根据消息解析 turnId 后触发重试 */
498
+ const handleRegenerate = useCallback(
499
+ async (msg: Message) => {
500
+ const turnId = resolveMessageTurnIdForFeedback(msg)
501
+ if (!turnId) return
502
+ const enableNetwork = getChatEnableNetwork()
503
+ const enableKnowledgeBase = getChatEnableKnowledgeBase()
504
+ const enableDataAnalysis = getChatEnableDataAnalysis()
505
+ await regenerateMessage(sessionId, turnId, {
506
+ enableNetwork,
507
+ enableKnowledgeBase,
508
+ enableDataAnalysis,
509
+ })
510
+ },
511
+ [regenerateMessage, sessionId],
512
+ )
513
+
514
+ /** 提交消息点赞/点踩:先乐观更新,再调用反馈接口,失败则回滚刷新 */
515
+ const handleFeedback = useCallback(
516
+ async (messageId: string, score: number) => {
517
+ const currentData = queryClient.getQueryData<Message[]>([
518
+ 'messages',
519
+ sessionId,
520
+ ])
521
+ const msg = currentData?.find((m) => m.id === messageId)
522
+ if (!msg) return
523
+
524
+ const display = getMessageEvaluationDisplay(msg)
525
+ const clickedLike = score === 1
526
+ const evaluationType = buildChatFeedbackEvaluationType(
527
+ display,
528
+ clickedLike,
529
+ )
530
+ const nextEvaluationType =
531
+ evaluationTypeAfterFeedbackRequest(evaluationType)
532
+
533
+ queryClient.setQueryData<Message[]>(['messages', sessionId], (old) => {
534
+ const prev = old ?? []
535
+ return prev.map((m) =>
536
+ m.id === messageId
537
+ ? {
538
+ ...m,
539
+ evaluation_type: nextEvaluationType,
540
+ feedback_score: undefined,
541
+ }
542
+ : m,
543
+ )
544
+ })
545
+
546
+ try {
547
+ await feedbackChatMessageAPI({
548
+ sessionId,
549
+ turnId: resolveMessageTurnIdForFeedback(msg),
550
+ evaluationType,
551
+ })
552
+ } catch (error) {
553
+ console.error('Feedback failed', error)
554
+ showToast(t('common.error'), 'error')
555
+ await queryClient.invalidateQueries({ queryKey: ['messages', sessionId] })
556
+ }
557
+ },
558
+ [sessionId, queryClient, showToast, t],
559
+ )
560
+
561
+ /** 当前会话是否处于流式生成中 */
562
+ const isCurrentThreadStreaming =
563
+ isStreaming && activeThreadId === sessionId
564
+ /** 是否有其他会话正在流式生成(用于禁用当前页部分操作) */
565
+ const isOtherThreadStreaming =
566
+ isStreaming && activeThreadId !== sessionId
567
+
568
+ /** 批量删除会话轮次消息,并刷新当前会话消息列表 */
569
+ const handleDeleteChatTurns = useCallback(
570
+ async (turnIds: string[]) => {
571
+ await deleteChatMessageAPI({
572
+ sessionId,
573
+ turnIdList: turnIds,
574
+ })
575
+ await queryClient.invalidateQueries({ queryKey: ['messages', sessionId] })
576
+ },
577
+ [queryClient, sessionId],
578
+ )
579
+
580
+ return (
581
+ <div className="flex h-full bg-white dark:bg-[#131314]">
582
+ <div className="flex-1 flex flex-col h-full relative">
583
+ {isDeletingThisSession && (
584
+ <div
585
+ className="absolute inset-0 z-20 flex items-center justify-center bg-white/75 backdrop-blur-[1px] dark:bg-[#131314]/80"
586
+ aria-busy="true"
587
+ aria-label={t('common.loading')}
588
+ >
589
+ <Loader2 size={28} className="animate-spin text-blue-500" />
590
+ </div>
591
+ )}
592
+ <ReferencesKnowledgeProvider key={decodedSessionId}>
593
+ <div className="flex-1 overflow-hidden flex flex-col pb-4">
594
+ <MessageList
595
+ key={sessionId}
596
+ messages={displayMessages}
597
+ isLoading={isCurrentThreadStreaming || isMessagesLoading}
598
+ hasMore={messagesHasMore}
599
+ isLoadingMore={isLoadingMoreMessages}
600
+ onLoadMore={handleLoadMoreMessages}
601
+ onOptionClick={handleSend}
602
+ onFeedback={handleFeedback}
603
+ onRegenerate={canResend && !isOtherThreadStreaming ? handleRegenerate : undefined}
604
+ enableMessageDelete={canDeleteMessage && !isOtherThreadStreaming}
605
+ onSelectionModeChange={setMessageSelectMode}
606
+ onDeleteTurns={handleDeleteChatTurns}
607
+ />
608
+ </div>
609
+ {chatPrepDemoEnabled && (
610
+ <div className="shrink-0 border-t border-gray-200/90 px-4 pt-2 dark:border-gray-700/90">
611
+ <div className="mx-auto flex w-full max-w-7xl justify-start">
612
+ <PreparationSteps items={CHAT_PREPARATION_DEMO_ITEMS} />
613
+ </div>
614
+ </div>
615
+ )}
616
+ <ReferencesKnowledgeAwareInput>
617
+ {canCreate && !messageSelectMode ? (
618
+ <ChatInput
619
+ onSend={handleSend}
620
+ onStop={handleStop}
621
+ isLoading={isCurrentThreadStreaming}
622
+ disableSend={isOtherThreadStreaming}
623
+ />
624
+ ) : !canCreate ? (
625
+ <div className="p-4 text-center text-gray-500 italic text-sm">
626
+ {t('chat.no_create_permission') ||
627
+ 'You do not have permission to send messages.'}
628
+ </div>
629
+ ) : null}
630
+ </ReferencesKnowledgeAwareInput>
631
+ </ReferencesKnowledgeProvider>
632
+ </div>
633
+ </div>
634
+ )
635
+ }
636
+
637
+ const ChatSessionPage: React.FC = () => {
638
+ /** 读取路由中的会话 id 参数 */
639
+ const { sessionId } = useParams<{ sessionId: string }>()
640
+ /** 会话 id 缺失时回退到会话首页 */
641
+ if (!sessionId) {
642
+ return <Navigate to="/chat" replace />
643
+ }
644
+ return <ChatSessionView sessionId={sessionId} />
645
+ }
646
+
647
+ export default ChatSessionPage