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,717 @@
1
+ import { useQuery } from '@tanstack/react-query';
2
+ import clsx from 'clsx';
3
+ import { Check, ChevronRight, Edit2, Loader2, Menu, MessageSquare, MoreVertical, Pin, PinOff, Plus, Search, Trash2, X } from 'lucide-react';
4
+ import React, { useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react';
5
+ import { useTranslation } from 'react-i18next';
6
+ import { Link, useLocation, useNavigate } from 'react-router-dom';
7
+
8
+ import { getChatSessionsAPI } from '@/api/chat';
9
+ import ConfirmDialog from '@/components/common/modal/confirm-dialog';
10
+ import { useChatContext } from '@/contexts/ChatContext';
11
+ import {
12
+ mergeStreamingPlaceholderThread,
13
+ normalizeSessionsToThreads,
14
+ parseChatSessionIdFromPathname,
15
+ resolveThreadDisplayTitle,
16
+ } from '@/utils/chatUtils';
17
+ import { getListPathFromSubRoutePathname } from '@/utils/menuRouteRegistry';
18
+ import {
19
+ useDeletingChatSessionId,
20
+ useThreadActions,
21
+ } from '../../../hooks/useThreadActions';
22
+ import type { Thread } from '../../../types';
23
+ import type { MenuTreeItem } from '../../../types/menu';
24
+ import { getMenuDisplayName } from '../../../utils/menuI18n';
25
+ import { filterMenuTree, getIconComponent } from '../../../utils/routeConfig';
26
+
27
+ /**
28
+ * 渲染管理菜单项(动态)
29
+ */
30
+ const renderAdminMenuItem = (
31
+ menu: MenuTreeItem,
32
+ location: any,
33
+ isCollapsed: boolean,
34
+ t: (key: string) => string,
35
+ ) => {
36
+ const Icon = getIconComponent(menu.icon);
37
+
38
+ // 如果component是Main,说明是父级菜单,不加载组件,没有实际跳转地址
39
+ const isParentMenu = menu.component === 'Main';
40
+
41
+ // 判断菜单是否激活:支持详情/提交页保持菜单激活状态
42
+ const currentPath = location.pathname;
43
+ const listPath = getListPathFromSubRoutePathname(currentPath);
44
+
45
+ let isActive = false;
46
+ if (menu.path && menu.path !== '/') {
47
+ // 提取菜单的基础路径(去掉 /index)
48
+ const menuBasePath = menu.path.replace(/\/index$/, '');
49
+
50
+ if (listPath) {
51
+ // 当前在详情/提交页,比较基础路径
52
+ const listBasePath = listPath.replace(/\/index$/, '');
53
+ isActive = menuBasePath === listBasePath;
54
+ } else {
55
+ // 普通页面,提取基础路径(去掉 /index、/detail/*、/submit/*)
56
+ const currentBasePath = currentPath.replace(/\/(index|detail|submit)(\/.*)?$/, '');
57
+ isActive = menuBasePath === currentBasePath;
58
+ }
59
+ }
60
+
61
+ // 如果有子菜单,渲染为可展开的菜单组
62
+ if (menu.children && menu.children.length > 0) {
63
+ return (
64
+ <div key={menu.id}>
65
+ <div
66
+ className={clsx(
67
+ 'flex items-center justify-between px-3 py-2 rounded-lg text-sm font-medium transition-colors',
68
+ 'text-gray-700 dark:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-700',
69
+ isCollapsed ? 'justify-center cursor-pointer' : 'cursor-default',
70
+ )}
71
+ title={isCollapsed ? getMenuDisplayName(menu, t) : undefined}
72
+ >
73
+ <div className="flex items-center gap-3">
74
+ {Icon && <Icon size={20} />}
75
+ <span className={clsx(isCollapsed ? 'hidden' : 'block')}>{getMenuDisplayName(menu, t)}</span>
76
+ </div>
77
+ </div>
78
+
79
+ {!isCollapsed && (
80
+ <div className="mt-1 ml-6 space-y-1">
81
+ {menu.children.map((child) => {
82
+ // 子菜单激活判断:支持详情/提交页
83
+ let isChildActive = false;
84
+ if (child.path) {
85
+ // 提取子菜单的基础路径
86
+ const childBasePath = child.path.replace(/\/index$/, '');
87
+
88
+ if (listPath) {
89
+ // 在详情/提交页,比较基础路径
90
+ const listBasePath = listPath.replace(/\/index$/, '');
91
+ isChildActive = childBasePath === listBasePath;
92
+ } else {
93
+ // 普通页面,提取基础路径(去掉 /index、/detail/*、/submit/*)
94
+ const currentBasePath = currentPath.replace(/\/(index|detail|submit)(\/.*)?$/, '');
95
+ isChildActive = childBasePath === currentBasePath;
96
+ }
97
+ }
98
+
99
+ return (
100
+ <Link
101
+ key={child.id}
102
+ to={child.path || '#'}
103
+ className={clsx(
104
+ 'flex items-center gap-3 px-3 py-1.5 rounded text-sm transition-colors',
105
+ isChildActive
106
+ ? 'bg-blue-50 dark:bg-blue-900/30 text-blue-700 dark:text-blue-300'
107
+ : 'hover:bg-gray-100 dark:hover:bg-gray-700 text-gray-700 dark:text-gray-300',
108
+ )}
109
+ >
110
+ {getIconComponent(child.icon) && React.createElement(getIconComponent(child.icon)!, { size: 16 })}
111
+ <span>{getMenuDisplayName(child, t)}</span>
112
+ </Link>
113
+ );
114
+ })}
115
+ </div>
116
+ )}
117
+ </div>
118
+ );
119
+ }
120
+
121
+ // 没有子菜单,渲染为普通链接
122
+ if (!menu.path || isParentMenu) return null;
123
+
124
+ return (
125
+ <Link
126
+ key={menu.id}
127
+ to={menu.path}
128
+ className={clsx(
129
+ 'flex items-center gap-3 px-3 py-2 rounded-lg text-sm font-medium transition-colors',
130
+ isActive
131
+ ? 'bg-blue-50 text-blue-700 dark:bg-blue-900/20 dark:text-blue-400'
132
+ : 'text-gray-700 dark:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-700',
133
+ isCollapsed && 'justify-center',
134
+ )}
135
+ title={isCollapsed ? getMenuDisplayName(menu, t) : undefined}
136
+ >
137
+ {Icon && <Icon size={20} />}
138
+ <span className={clsx(isCollapsed ? 'hidden' : 'block')}>{getMenuDisplayName(menu, t)}</span>
139
+ </Link>
140
+ );
141
+ };
142
+
143
+ type SidebarProps = {
144
+ user: any;
145
+ hasPermission: (permission: string) => boolean;
146
+ menus: MenuTreeItem[];
147
+ isLoadingMenus: boolean;
148
+ isCollapsed: boolean;
149
+ setIsCollapsed: React.Dispatch<React.SetStateAction<boolean>>;
150
+ isAdminMenuOpen: boolean;
151
+ setIsAdminMenuOpen: React.Dispatch<React.SetStateAction<boolean>>;
152
+ };
153
+
154
+ const Sidebar: React.FC<SidebarProps> = ({
155
+ user,
156
+ hasPermission,
157
+ menus,
158
+ isLoadingMenus,
159
+ isCollapsed,
160
+ setIsCollapsed,
161
+ isAdminMenuOpen,
162
+ setIsAdminMenuOpen,
163
+ }) => {
164
+ const { t } = useTranslation();
165
+ const location = useLocation();
166
+ const navigate = useNavigate();
167
+ const { activeThreadId, isStreaming, streamingPlaceholderThread } =
168
+ useChatContext();
169
+
170
+ const canChatSessions = hasPermission('system:chat:sessions');
171
+ const canChatSend = hasPermission('system:chat:send');
172
+ const canSessionRename = hasPermission('system:chat:session-update-title');
173
+ const canSessionDelete = hasPermission('system:chat:session-delete');
174
+ const canSessionTop =
175
+ hasPermission('system:chat:session-top') ||
176
+ hasPermission('system:chat:session-cancel-top');
177
+
178
+ const [searchQuery, setSearchQuery] = useState('');
179
+ const [isSearchVisible, setIsSearchVisible] = useState(false);
180
+ const [extraThreads, setExtraThreads] = useState<Thread[]>([]);
181
+ const [sessionsHasMore, setSessionsHasMore] = useState(true);
182
+ const [sessionsLoadingMore, setSessionsLoadingMore] = useState(false);
183
+ const navScrollRef = useRef<HTMLElement | null>(null);
184
+ /** 会话列表向下加载更多:保持当前滚动位置,避免列表更新后跳到顶部 */
185
+ const sessionsScrollSnapshotRef = useRef<{
186
+ scrollTop: number;
187
+ } | null>(null);
188
+
189
+ // Context Menu State (chat threads)
190
+ const [activeMenuId, setActiveMenuId] = useState<string | null>(null);
191
+ const [editThreadId, setEditThreadId] = useState<string | null>(null);
192
+ const [renameTitle, setRenameTitle] = useState('');
193
+ const [deleteConfirmId, setDeleteConfirmId] = useState<string | null>(null);
194
+
195
+ /** 管理菜单根节点(含按钮与浮层);用于捕获阶段判断「点在外面」 */
196
+ const adminMenuRootRef = useRef<HTMLDivElement | null>(null);
197
+
198
+ // 会话行「更多」菜单:依赖 window click(会话行内无 stopPropagation 拦到 window)
199
+ useEffect(() => {
200
+ if (!activeMenuId) return;
201
+ const handleClickOutside = () => setActiveMenuId(null);
202
+ window.addEventListener('click', handleClickOutside);
203
+ return () => window.removeEventListener('click', handleClickOutside);
204
+ }, [activeMenuId]);
205
+
206
+ // 管理菜单:Header 等区域对 click 使用了 stopPropagation,冒泡到 window 收不到;用 document mousedown 捕获 + contains
207
+ useEffect(() => {
208
+ if (!isAdminMenuOpen) return;
209
+ const onDocPointerDown = (e: MouseEvent) => {
210
+ const root = adminMenuRootRef.current;
211
+ if (root && !root.contains(e.target as Node)) {
212
+ setIsAdminMenuOpen(false);
213
+ }
214
+ };
215
+ document.addEventListener('mousedown', onDocPointerDown, true);
216
+ return () => document.removeEventListener('mousedown', onDocPointerDown, true);
217
+ }, [isAdminMenuOpen, setIsAdminMenuOpen]);
218
+
219
+ // 管理弹层只展示第一个根菜单下的子菜单(如「菜单」下的 Web 菜单),
220
+ // 不再把所有根节点一起展开,避免出现额外的一层「菜单」包裹。
221
+ const filteredDynamicMenus = useMemo(() => {
222
+ if (!menus || menus.length === 0) return [];
223
+ const roots = menus[0]?.children || [];
224
+ return filterMenuTree(roots, hasPermission);
225
+ }, [menus, hasPermission]);
226
+
227
+ const { data: threadsFirstPage = [], isLoading: isLoadingThreads } = useQuery({
228
+ queryKey: ['threads'],
229
+ queryFn: async () => {
230
+ const raw = await getChatSessionsAPI();
231
+ const list = normalizeSessionsToThreads(raw);
232
+ setExtraThreads([]);
233
+ setSessionsHasMore(list.length > 0);
234
+ return list;
235
+ },
236
+ enabled: !!user && canChatSessions,
237
+ });
238
+
239
+ const threadsRemote = useMemo(
240
+ () => [...threadsFirstPage, ...extraThreads],
241
+ [threadsFirstPage, extraThreads],
242
+ );
243
+
244
+ const loadMoreSessions = async () => {
245
+ if (sessionsLoadingMore || !sessionsHasMore || !canChatSessions) return;
246
+ const all = [...threadsFirstPage, ...extraThreads];
247
+ const last = all[all.length - 1];
248
+ const createTime = last?.created_at?.trim();
249
+ if (!createTime) {
250
+ setSessionsHasMore(false);
251
+ return;
252
+ }
253
+ const navEl = navScrollRef.current;
254
+ if (navEl) {
255
+ sessionsScrollSnapshotRef.current = { scrollTop: navEl.scrollTop };
256
+ }
257
+ setSessionsLoadingMore(true);
258
+ try {
259
+ const raw = await getChatSessionsAPI({ createTime });
260
+ const batch = normalizeSessionsToThreads(raw);
261
+ if (batch.length === 0) {
262
+ setSessionsHasMore(false);
263
+ } else {
264
+ setExtraThreads((prev) => {
265
+ const ids = new Set([...threadsFirstPage, ...prev].map((t) => t.id));
266
+ return [...prev, ...batch.filter((t) => !ids.has(t.id))];
267
+ });
268
+ }
269
+ } catch (e) {
270
+ console.error('[Sidebar] load more sessions failed', e);
271
+ } finally {
272
+ setSessionsLoadingMore(false);
273
+ }
274
+ };
275
+
276
+ const handleNavScroll = () => {
277
+ const el = navScrollRef.current;
278
+ if (!el || sessionsLoadingMore || !sessionsHasMore || isLoadingThreads) return;
279
+ const gap = el.scrollHeight - el.scrollTop - el.clientHeight;
280
+ if (gap < 80) {
281
+ void loadMoreSessions();
282
+ }
283
+ };
284
+
285
+ /** 流式进行中:接口会话列表若尚未含新会话,合并乐观占位行,避免左侧被空数据顶掉 */
286
+ const threads = useMemo(
287
+ () =>
288
+ mergeStreamingPlaceholderThread(threadsRemote, {
289
+ placeholder: streamingPlaceholderThread,
290
+ activeThreadId,
291
+ isStreaming,
292
+ }),
293
+ [threadsRemote, streamingPlaceholderThread, activeThreadId, isStreaming],
294
+ );
295
+
296
+ const filteredThreads = useMemo(() => {
297
+ if (!searchQuery.trim()) return threads;
298
+ return threads.filter((thread) =>
299
+ (thread.title || '').toLowerCase().includes(searchQuery.toLowerCase()),
300
+ );
301
+ }, [threads, searchQuery]);
302
+
303
+ const pathChatSessionId = useMemo(
304
+ () => parseChatSessionIdFromPathname(location.pathname),
305
+ [location.pathname],
306
+ );
307
+
308
+ /** 仅切换会话时滚到可见;列表加载更多时不要因 filteredThreads 变化而 scrollIntoView */
309
+ useEffect(() => {
310
+ if (!pathChatSessionId || isLoadingThreads) return;
311
+ const el = document.querySelector(
312
+ `[data-chat-thread-id="${CSS.escape(pathChatSessionId)}"]`,
313
+ );
314
+ if (el instanceof HTMLElement) {
315
+ el.scrollIntoView({ block: 'nearest', behavior: 'smooth' });
316
+ }
317
+ }, [pathChatSessionId, isLoadingThreads]);
318
+
319
+ useLayoutEffect(() => {
320
+ const navEl = navScrollRef.current;
321
+ const snap = sessionsScrollSnapshotRef.current;
322
+ if (!navEl || !snap || sessionsLoadingMore) return;
323
+ navEl.scrollTop = snap.scrollTop;
324
+ sessionsScrollSnapshotRef.current = null;
325
+ }, [sessionsLoadingMore, extraThreads]);
326
+
327
+ const deletingSessionId = useDeletingChatSessionId();
328
+
329
+ const { renameThread, togglePinThread, deleteThread } = useThreadActions({
330
+ onDeleteSuccess: (threadId) => {
331
+ const m = location.pathname.match(/^\/chat\/([^/]+)\/?$/);
332
+ const currentSessionId = m?.[1] ? decodeURIComponent(m[1]) : null;
333
+ if (currentSessionId && currentSessionId === threadId) {
334
+ navigate('/chat');
335
+ }
336
+ },
337
+ onUpdateSuccess: () => {
338
+ setEditThreadId(null);
339
+ },
340
+ });
341
+
342
+ const handleDeleteThread = (e: React.MouseEvent, threadId: string) => {
343
+ e.stopPropagation();
344
+ setActiveMenuId(null);
345
+ setDeleteConfirmId(threadId);
346
+ };
347
+
348
+ const handleStartRename = (e: React.MouseEvent, thread: Thread) => {
349
+ e.stopPropagation();
350
+ setActiveMenuId(null);
351
+ setEditThreadId(thread.id);
352
+ setRenameTitle(thread.title);
353
+ };
354
+
355
+ const handlePin = (e: React.MouseEvent, thread: Thread) => {
356
+ e.stopPropagation();
357
+ setActiveMenuId(null);
358
+ togglePinThread(thread.id, thread.top_status === 1);
359
+ };
360
+
361
+ const handleMenuClick = (e: React.MouseEvent, id: string) => {
362
+ e.stopPropagation();
363
+ e.nativeEvent.stopImmediatePropagation();
364
+ setActiveMenuId(activeMenuId === id ? null : id);
365
+ };
366
+
367
+ const handleSaveRename = (threadId: string) => {
368
+ if (renameTitle.trim()) {
369
+ renameThread(threadId, renameTitle);
370
+ } else {
371
+ setEditThreadId(null);
372
+ }
373
+ };
374
+
375
+ const handleNewChat = () => {
376
+ navigate('/chat');
377
+ };
378
+
379
+ const handleSelectThread = (threadId: string) => {
380
+ navigate(`/chat/${encodeURIComponent(threadId)}`);
381
+ };
382
+
383
+ const onConfirmDelete = () => {
384
+ if (deleteConfirmId) {
385
+ deleteThread(deleteConfirmId);
386
+ setDeleteConfirmId(null);
387
+ }
388
+ };
389
+
390
+ return (
391
+ <>
392
+ <ConfirmDialog
393
+ isOpen={!!deleteConfirmId}
394
+ onClose={() => setDeleteConfirmId(null)}
395
+ onConfirm={onConfirmDelete}
396
+ title={t('chat.delete_title')}
397
+ message={t('chat.delete_confirm')}
398
+ confirmText={t('common.delete')}
399
+ danger
400
+ />
401
+
402
+ <aside
403
+ className={clsx(
404
+ 'bg-white dark:bg-gray-800 border-r border-gray-200 dark:border-gray-700 flex flex-col transition-all duration-300 ease-in-out relative z-30',
405
+ isCollapsed ? 'w-16' : 'w-64',
406
+ )}
407
+ >
408
+ <div className={clsx('p-3 flex items-center', isCollapsed ? 'justify-center' : 'justify-between')}>
409
+ <button
410
+ onClick={() => {
411
+ if (isCollapsed) {
412
+ setIsCollapsed(false);
413
+ } else {
414
+ setIsCollapsed(true);
415
+ setIsSearchVisible(false);
416
+ }
417
+ }}
418
+ className="p-2 text-gray-500 hover:bg-gray-100 dark:hover:bg-gray-700 rounded-lg transition-colors"
419
+ title={isCollapsed ? t('common.expand') : t('common.collapse')}
420
+ >
421
+ <Menu size={20} />
422
+ </button>
423
+
424
+ {!isCollapsed && (
425
+ <button
426
+ onClick={() => {
427
+ setIsSearchVisible(!isSearchVisible);
428
+ if (isSearchVisible) {
429
+ setSearchQuery('');
430
+ }
431
+ }}
432
+ className={clsx(
433
+ 'p-2 rounded-lg transition-colors',
434
+ isSearchVisible
435
+ ? 'bg-blue-50 text-blue-600 dark:bg-blue-900/30 dark:text-blue-400'
436
+ : 'text-gray-500 hover:bg-gray-100 dark:hover:bg-gray-700',
437
+ )}
438
+ title={t('common.search_chats')}
439
+ type="button"
440
+ >
441
+ {isSearchVisible ? <X size={20} /> : <Search size={20} />}
442
+ </button>
443
+ )}
444
+ </div>
445
+
446
+ {!isCollapsed && isSearchVisible && (
447
+ <div className="px-3 pb-2 relative">
448
+ <input
449
+ type="text"
450
+ value={searchQuery}
451
+ onChange={(e) => setSearchQuery(e.target.value)}
452
+ placeholder={t('common.search_placeholder')}
453
+ className="w-full pl-3 pr-8 py-1.5 text-sm bg-gray-100 dark:bg-gray-700/50 border-none rounded-md focus:ring-1 focus:ring-blue-500 text-gray-900 dark:text-gray-100 placeholder-gray-500"
454
+ autoFocus
455
+ />
456
+ {searchQuery && (
457
+ <button
458
+ onClick={() => setSearchQuery('')}
459
+ className="absolute right-4 top-1.5 p-0.5 text-gray-400 hover:text-gray-600 dark:hover:text-gray-200 transition-colors"
460
+ >
461
+ <X size={14} />
462
+ </button>
463
+ )}
464
+ </div>
465
+ )}
466
+
467
+ {canChatSend && (
468
+ <div className="px-3 pb-2">
469
+ <button
470
+ onClick={handleNewChat}
471
+ className={clsx(
472
+ 'w-full flex items-center justify-center gap-2 px-3 py-2 rounded-lg transition-all duration-200 border',
473
+ 'bg-transparent border-gray-200 dark:border-gray-700 text-gray-500 dark:text-gray-400',
474
+ 'hover:border-blue-300 dark:hover:border-blue-700 hover:text-blue-600 dark:hover:text-blue-400 hover:bg-blue-50 dark:hover:bg-blue-900/10',
475
+ isCollapsed ? 'px-0 aspect-square' : '',
476
+ )}
477
+ title={t('chat.new_chat')}
478
+ >
479
+ <Plus size={isCollapsed ? 20 : 18} />
480
+ {!isCollapsed && <span className="font-medium text-sm">{t('chat.new_chat')}</span>}
481
+ </button>
482
+ </div>
483
+ )}
484
+
485
+ <nav
486
+ ref={navScrollRef}
487
+ onScroll={handleNavScroll}
488
+ className="flex-1 px-3 space-y-1 overflow-y-auto overflow-x-hidden"
489
+ >
490
+ {/* Chat section */}
491
+ <div>
492
+ <div
493
+ className={clsx(
494
+ 'flex items-center justify-between px-3 py-2 rounded-lg text-sm font-medium transition-colors',
495
+ 'text-gray-700 dark:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-700',
496
+ isCollapsed ? 'justify-center cursor-pointer' : 'cursor-default',
497
+ )}
498
+ onClick={() => {
499
+ if (isCollapsed) {
500
+ setIsCollapsed(false);
501
+ }
502
+ }}
503
+ title={isCollapsed ? t('common.chat') : undefined}
504
+ >
505
+ <div className="flex items-center gap-3">
506
+ <MessageSquare size={20} />
507
+ <span className={clsx(isCollapsed ? 'hidden' : 'block')}>{t('common.chat')}</span>
508
+ </div>
509
+ </div>
510
+
511
+ {!isCollapsed && canChatSessions && (
512
+ <div className="mt-1 ml-6 space-y-1">
513
+ {isLoadingThreads ? (
514
+ <div className="flex justify-center py-2">
515
+ <Loader2 size={16} className="animate-spin text-gray-400" />
516
+ </div>
517
+ ) : (
518
+ filteredThreads.map((thread) => {
519
+ const isOnChatWelcome =
520
+ location.pathname === '/chat' ||
521
+ location.pathname === '/chat/';
522
+ const isActive =
523
+ thread.id === pathChatSessionId ||
524
+ (isOnChatWelcome &&
525
+ isStreaming &&
526
+ activeThreadId != null &&
527
+ activeThreadId === thread.id);
528
+ const isEditing = editThreadId === thread.id;
529
+ return (
530
+ <div
531
+ key={thread.id}
532
+ data-chat-thread-id={thread.id}
533
+ onClick={() => !isEditing && handleSelectThread(thread.id)}
534
+ className={clsx(
535
+ 'group relative flex items-center justify-between px-3 py-1.5 rounded text-sm cursor-pointer transition-colors',
536
+ isActive
537
+ ? 'bg-blue-100 dark:bg-blue-900/30 text-blue-700 dark:text-blue-300'
538
+ : 'hover:bg-gray-100 dark:hover:bg-gray-700 text-gray-600 dark:text-gray-400',
539
+ )}
540
+ >
541
+ {isEditing ? (
542
+ <div className="flex items-center gap-1 w-full" onClick={(e) => e.stopPropagation()}>
543
+ <input
544
+ type="text"
545
+ value={renameTitle}
546
+ onChange={(e) => setRenameTitle(e.target.value)}
547
+ onKeyDown={(e) => {
548
+ if (e.key === 'Enter') handleSaveRename(thread.id);
549
+ if (e.key === 'Escape') setEditThreadId(null);
550
+ }}
551
+ className="flex-1 bg-white dark:bg-gray-900 border border-blue-300 dark:border-blue-700 rounded px-1 py-0.5 text-xs focus:outline-none focus:ring-1 focus:ring-blue-500"
552
+ autoFocus
553
+ />
554
+ <button onClick={() => handleSaveRename(thread.id)} className="p-0.5 text-green-600 hover:bg-green-100 rounded">
555
+ <Check size={12} />
556
+ </button>
557
+ <button onClick={() => setEditThreadId(null)} className="p-0.5 text-gray-500 hover:bg-gray-200 rounded">
558
+ <X size={12} />
559
+ </button>
560
+ </div>
561
+ ) : (
562
+ <>
563
+ <div className="flex items-center gap-2 overflow-hidden flex-1">
564
+ {thread.top_status === 1 && (
565
+ <Pin
566
+ size={10}
567
+ className="shrink-0 fill-current text-blue-500 rotate-45"
568
+ />
569
+ )}
570
+ <span
571
+ className={clsx(
572
+ 'truncate flex-1',
573
+ thread.top_status === 1 &&
574
+ 'font-medium text-blue-600 dark:text-blue-400',
575
+ )}
576
+ >
577
+ {resolveThreadDisplayTitle(
578
+ thread.title,
579
+ t('chat.untitled_session'),
580
+ )}
581
+ </span>
582
+ </div>
583
+
584
+ <div className="relative shrink-0 ml-1">
585
+ <button
586
+ onClick={(e) => handleMenuClick(e, thread.id)}
587
+ className={clsx(
588
+ 'p-0.5 rounded-full hover:bg-gray-200 dark:hover:bg-gray-600 transition-opacity',
589
+ activeMenuId === thread.id || 'group-hover:opacity-100 opacity-0',
590
+ )}
591
+ >
592
+ <MoreVertical size={14} />
593
+ </button>
594
+
595
+ {activeMenuId === thread.id && (
596
+ <div
597
+ className="absolute right-0 top-6 z-50 w-32 bg-white dark:bg-gray-900 shadow-xl rounded-lg border border-gray-100 dark:border-gray-700 py-1 overflow-hidden"
598
+ onClick={(e) => e.stopPropagation()}
599
+ >
600
+ {canSessionTop && (
601
+ <button
602
+ onClick={(e) => handlePin(e, thread)}
603
+ className="w-full text-left px-3 py-2 text-xs flex items-center gap-2 hover:bg-gray-100 dark:hover:bg-gray-800 text-gray-700 dark:text-gray-300"
604
+ >
605
+ {thread.top_status === 1 ? (
606
+ <PinOff size={12} />
607
+ ) : (
608
+ <Pin size={12} />
609
+ )}
610
+ <span>
611
+ {thread.top_status === 1
612
+ ? t('common.unpin')
613
+ : t('common.pin')}
614
+ </span>
615
+ </button>
616
+ )}
617
+ {canSessionRename && (
618
+ <button
619
+ onClick={(e) => handleStartRename(e, thread)}
620
+ className="w-full text-left px-3 py-2 text-xs flex items-center gap-2 hover:bg-gray-100 dark:hover:bg-gray-800 text-gray-700 dark:text-gray-300"
621
+ >
622
+ <Edit2 size={12} />
623
+ <span>{t('common.rename')}</span>
624
+ </button>
625
+ )}
626
+ {canSessionDelete && (
627
+ <>
628
+ <div className="h-px bg-gray-100 dark:bg-gray-800 my-1" />
629
+ <button
630
+ onClick={(e) => handleDeleteThread(e, thread.id)}
631
+ className="w-full text-left px-3 py-2 text-xs flex items-center gap-2 hover:bg-red-50 dark:hover:bg-red-900/10 text-red-600 dark:text-red-400"
632
+ >
633
+ <Trash2 size={12} />
634
+ <span>{t('common.delete')}</span>
635
+ </button>
636
+ </>
637
+ )}
638
+ </div>
639
+ )}
640
+ </div>
641
+ </>
642
+ )}
643
+ {deletingSessionId === thread.id && (
644
+ <div
645
+ className="absolute inset-0 z-[60] flex items-center justify-center rounded bg-white/80 backdrop-blur-[1px] dark:bg-gray-900/75"
646
+ aria-busy="true"
647
+ aria-label={t('common.loading')}
648
+ >
649
+ <Loader2
650
+ size={18}
651
+ className="animate-spin text-blue-500"
652
+ />
653
+ </div>
654
+ )}
655
+ </div>
656
+ );
657
+ })
658
+ )}
659
+ {!isLoadingThreads && filteredThreads.length === 0 && searchQuery && (
660
+ <div className="px-3 py-2 text-xs text-gray-400 text-center italic">
661
+ {t('common.no_matches')}
662
+ </div>
663
+ )}
664
+ </div>
665
+ )}
666
+ </div>
667
+ </nav>
668
+
669
+ {/* 动态管理菜单 - 与会话列表拉开间距,减少误触 */}
670
+ {isLoadingMenus ? (
671
+ <div className="mt-4 shrink-0 px-3 py-4 flex justify-center">
672
+ <Loader2 size={20} className="animate-spin text-gray-400" />
673
+ </div>
674
+ ) : filteredDynamicMenus.length > 0 ? (
675
+ <div ref={adminMenuRootRef} className="relative mt-4 shrink-0">
676
+ <button
677
+ onClick={(e) => {
678
+ e.stopPropagation();
679
+ setIsAdminMenuOpen(!isAdminMenuOpen);
680
+ }}
681
+ className={clsx(
682
+ 'flex items-center justify-between w-full px-3 py-2 rounded-lg text-sm font-medium transition-colors',
683
+ isAdminMenuOpen
684
+ ? 'bg-blue-50 text-blue-700 dark:bg-blue-900/20 dark:text-blue-400'
685
+ : 'text-gray-700 dark:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-700',
686
+ isCollapsed && 'justify-center',
687
+ )}
688
+ title={isCollapsed ? t('common.admin') : undefined}
689
+ >
690
+ <div className="flex items-center gap-3">
691
+ {getIconComponent('Shield') && React.createElement(getIconComponent('Shield')!, { size: 20 })}
692
+ <span className={clsx(isCollapsed ? 'hidden' : 'block')}>{t('common.admin')}</span>
693
+ </div>
694
+ {/* className={clsx('transition-transform', isAdminMenuOpen && 'rotate-90')}*/}
695
+ {!isCollapsed && <ChevronRight size={16} />}
696
+ </button>
697
+
698
+ {isAdminMenuOpen && !isCollapsed && (
699
+ <div className="absolute left-full bottom-0 ml-2 w-56 bg-white dark:bg-gray-800 shadow-2xl rounded-xl border border-gray-100 dark:border-gray-700 overflow-hidden z-50">
700
+ <div className="p-2">
701
+ {filteredDynamicMenus.map((menu) => (
702
+ <div key={menu.id} className="space-y-1">
703
+ {renderAdminMenuItem(menu, location, false, t)}
704
+ </div>
705
+ ))}
706
+ </div>
707
+ </div>
708
+ )}
709
+ </div>
710
+ ) : null}
711
+ </aside>
712
+ </>
713
+ );
714
+ };
715
+
716
+ export default Sidebar;
717
+