cx-chat 0.0.2 → 0.0.4

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 (406) hide show
  1. package/dist/cx-chat.css +1 -0
  2. package/dist/cx-chat.es.js +68257 -0
  3. package/dist/cx-chat.umd.js +492 -0
  4. package/package.json +27 -3
  5. package/.cursor/rules/i18n-cn-gloss-comments.mdc +0 -31
  6. package/.cursor/rules/list-page-view-pageconfig.mdc +0 -32
  7. package/.cursor/rules/no-over-defensive-programming.mdc +0 -90
  8. package/.cursor/rules/requirement-description-for-agent.mdc +0 -33
  9. package/.cursor/rules/use-showToast-not-antd-message.mdc +0 -28
  10. package/.docker/Dockerfile +0 -7
  11. package/.env +0 -9
  12. package/.env.development +0 -7
  13. package/.env.production +0 -7
  14. package/.gitlab-ci/docker-build.yaml +0 -28
  15. package/.gitlab-ci/k8s-deploy-dev-master.yaml +0 -42
  16. package/.gitlab-ci/npm-build.yaml +0 -17
  17. package/.gitlab-ci.yml +0 -8
  18. package/.k8s/0-namespace.yaml +0 -6
  19. package/.k8s/1-configmap-web.yaml +0 -7
  20. package/.k8s/1-nginx-conf-dev.yaml +0 -110
  21. package/.k8s/2-deployment.yaml +0 -27
  22. package/.k8s/3-service.yaml +0 -16
  23. package/.k8s/4-ingress-dev.yaml +0 -30
  24. package/.lingma/rules/use-showToast-not-antd-message.md +0 -34
  25. package/.nginx/nginx.conf +0 -52
  26. package/.prettierrc +0 -9
  27. package/eslint.config.js +0 -32
  28. package/index.html +0 -13
  29. package/postcss.config.js +0 -6
  30. package/src/App.tsx +0 -96
  31. package/src/_doc/0.docs-overview.md +0 -28
  32. package/src/_doc/cx-ui/0.docs-overview.md +0 -30
  33. package/src/_doc/cx-ui/comp.1.cx-ui-overview.md +0 -82
  34. package/src/_doc/cx-ui/comp.2.cx-modal.md +0 -82
  35. package/src/_doc/cx-ui/comp.3.cx-button.md +0 -89
  36. package/src/_doc/cx-ui/comp.4.cx-form.md +0 -72
  37. package/src/_doc/cx-ui/comp.5.cx-fields.md +0 -76
  38. package/src/_doc/cx-ui/comp.6.cx-tag.md +0 -57
  39. package/src/_doc/cx-ui/comp.7.cx-empty-state.md +0 -29
  40. package/src/_doc/meta/0.docs-overview.md +0 -24
  41. package/src/_doc/meta/comp.1.enum-runtime.md +0 -33
  42. package/src/_doc/meta/comp.2.dict-runtime.md +0 -39
  43. package/src/_doc/router/0.docs-overview.md +0 -14
  44. package/src/_doc/router/guide.1.menu-component-config.md +0 -181
  45. package/src/_doc/router/guide.2.router-auto-registration.md +0 -114
  46. package/src/_doc/table-view/0.docs-overview.md +0 -30
  47. package/src/_doc/table-view/comp.1.table-view.md +0 -542
  48. package/src/_doc/table-view/props.1.create-table-view-config.md +0 -193
  49. package/src/_doc/table-view/props.2.table-view-search-fields.md +0 -106
  50. package/src/api/_mock/README.md +0 -340
  51. package/src/api/_mock/api.ts +0 -1642
  52. package/src/api/_mock/bundle-shim.ts +0 -16
  53. package/src/api/_mock/handler-shim.ts +0 -6
  54. package/src/api/_mock/handler.ts +0 -458
  55. package/src/api/_mock/index.ts +0 -711
  56. package/src/api/_mock/interceptor.ts +0 -15
  57. package/src/api/_mock/mod.ts +0 -12
  58. package/src/api/_mock/utils.ts +0 -65
  59. package/src/api/base/memory.js +0 -24
  60. package/src/api/chat.js +0 -210
  61. package/src/api/common/auth.js +0 -70
  62. package/src/api/menus/business-rules.js +0 -76
  63. package/src/api/menus/feedback.js +0 -102
  64. package/src/api/menus/knowledge.js +0 -159
  65. package/src/api/menus/model-metadata/manage.js +0 -70
  66. package/src/api/menus/model-metadata/role.js +0 -50
  67. package/src/api/menus/model-metadata/training-detail-mock-data.js +0 -569
  68. package/src/api/menus/model-metadata/training.js +0 -28
  69. package/src/api/menus/skill.js +0 -40
  70. package/src/api/system/agent-config.js +0 -16
  71. package/src/api/system/department.js +0 -94
  72. package/src/api/system/dict.js +0 -86
  73. package/src/api/system/menu.js +0 -37
  74. package/src/api/system/permission.js +0 -26
  75. package/src/api/system/role.js +0 -34
  76. package/src/api/system/sys-config.js +0 -16
  77. package/src/api/system/sys-log.js +0 -17
  78. package/src/api/system/user.js +0 -75
  79. package/src/api/upload.js +0 -39
  80. package/src/assets/react.svg +0 -1
  81. package/src/components/auth/current-user-avatar.tsx +0 -77
  82. package/src/components/common/code-view.tsx +0 -149
  83. package/src/components/common/detail-link.tsx +0 -67
  84. package/src/components/common/error-boundary.tsx +0 -98
  85. package/src/components/common/language-switcher.tsx +0 -91
  86. package/src/components/common/lite-table/index.tsx +0 -135
  87. package/src/components/common/md-editor.tsx +0 -126
  88. package/src/components/common/modal/confirm-dialog.tsx +0 -113
  89. package/src/components/common/modal/dep-user-select-multi.tsx +0 -324
  90. package/src/components/common/modal/dep-user-select.tsx +0 -249
  91. package/src/components/common/modal/user-select-multi.tsx +0 -266
  92. package/src/components/common/pagination.tsx +0 -472
  93. package/src/components/common/path.tsx +0 -175
  94. package/src/components/common/system-logo-mark.tsx +0 -48
  95. package/src/components/cx-ui/button/index.less +0 -208
  96. package/src/components/cx-ui/button/index.tsx +0 -611
  97. package/src/components/cx-ui/checkbox/index.tsx +0 -78
  98. package/src/components/cx-ui/date-picker/index.less +0 -17
  99. package/src/components/cx-ui/date-picker/index.tsx +0 -193
  100. package/src/components/cx-ui/drawer/index.tsx +0 -47
  101. package/src/components/cx-ui/empty-state/index.tsx +0 -20
  102. package/src/components/cx-ui/floating-shell/CxFloatingShell.tsx +0 -89
  103. package/src/components/cx-ui/floating-shell/cx-floating-shell.less +0 -283
  104. package/src/components/cx-ui/floating-shell/has-floating-value.ts +0 -41
  105. package/src/components/cx-ui/form/CxForm.tsx +0 -15
  106. package/src/components/cx-ui/form/index.tsx +0 -20
  107. package/src/components/cx-ui/form-item/index.less +0 -26
  108. package/src/components/cx-ui/form-item/index.tsx +0 -36
  109. package/src/components/cx-ui/index.ts +0 -70
  110. package/src/components/cx-ui/input/auto-complete.tsx +0 -134
  111. package/src/components/cx-ui/input/index.tsx +0 -259
  112. package/src/components/cx-ui/input-number/index.jsx +0 -66
  113. package/src/components/cx-ui/modal/index.jsx +0 -212
  114. package/src/components/cx-ui/modal/index.less +0 -144
  115. package/src/components/cx-ui/modal/useCxModal.ts +0 -125
  116. package/src/components/cx-ui/multi-select/index.jsx +0 -74
  117. package/src/components/cx-ui/multi-select/index.less +0 -40
  118. package/src/components/cx-ui/multi-select/index2.tsx +0 -361
  119. package/src/components/cx-ui/radio/index.tsx +0 -33
  120. package/src/components/cx-ui/range-picker/index.less +0 -65
  121. package/src/components/cx-ui/range-picker/index.tsx +0 -219
  122. package/src/components/cx-ui/select/index.less +0 -34
  123. package/src/components/cx-ui/select/index.tsx +0 -196
  124. package/src/components/cx-ui/skeleton/index.tsx +0 -12
  125. package/src/components/cx-ui/steps/index.tsx +0 -14
  126. package/src/components/cx-ui/styles/_tokens.less +0 -79
  127. package/src/components/cx-ui/styles/index.less +0 -246
  128. package/src/components/cx-ui/switch/index.less +0 -106
  129. package/src/components/cx-ui/switch/index.tsx +0 -120
  130. package/src/components/cx-ui/table/index.less +0 -160
  131. package/src/components/cx-ui/table/index.tsx +0 -152
  132. package/src/components/cx-ui/tabs/index.less +0 -15
  133. package/src/components/cx-ui/tabs/index.tsx +0 -34
  134. package/src/components/cx-ui/tag/index.less +0 -51
  135. package/src/components/cx-ui/tag/index.tsx +0 -140
  136. package/src/components/cx-ui/timeline/index.tsx +0 -14
  137. package/src/components/cx-ui/tooltip/index.tsx +0 -67
  138. package/src/components/cx-ui/tree/index.tsx +0 -193
  139. package/src/components/cx-ui/tree-select/index.jsx +0 -91
  140. package/src/components/cx-ui/tree-select/index.less +0 -27
  141. package/src/components/cx-ui/upload-file/index.less +0 -223
  142. package/src/components/cx-ui/upload-file/index.tsx +0 -640
  143. package/src/components/cx-ui/upload-img/index.tsx +0 -291
  144. package/src/components/layout/components/Header.tsx +0 -216
  145. package/src/components/layout/components/Sidebar.tsx +0 -717
  146. package/src/components/layout/index.tsx +0 -95
  147. package/src/components/table-view/components/search-area.tsx +0 -411
  148. package/src/components/table-view/components/table-view-config.tsx +0 -528
  149. package/src/components/table-view/components/table-view.types.ts +0 -478
  150. package/src/components/table-view/components/tree-api-normalize.ts +0 -38
  151. package/src/components/table-view/components/tree-data-annotate.ts +0 -31
  152. package/src/components/table-view/components/tree-sidebar.tsx +0 -74
  153. package/src/components/table-view/index.tsx +0 -61
  154. package/src/components/table-view/list-page-view.tsx +0 -1049
  155. package/src/components/table-view/select-table-view.tsx +0 -1094
  156. package/src/components/table-view/styles/select-table-view.less +0 -51
  157. package/src/config/default-system-name.ts +0 -9
  158. package/src/config/system.ts +0 -165
  159. package/src/constants/countryCodes.ts +0 -3
  160. package/src/contexts/AuthContext.tsx +0 -256
  161. package/src/contexts/ChatContext.tsx +0 -839
  162. package/src/contexts/MenuContext.tsx +0 -62
  163. package/src/contexts/ToastContext.tsx +0 -181
  164. package/src/hooks/useCopyToClipboard.ts +0 -47
  165. package/src/hooks/useModalSubmit.ts +0 -104
  166. package/src/hooks/useRouter.ts +0 -240
  167. package/src/hooks/useStepForm.ts +0 -46
  168. package/src/hooks/useStickyHeader.ts +0 -42
  169. package/src/hooks/useThreadActions.ts +0 -105
  170. package/src/hooks/useUserPreferences.ts +0 -101
  171. package/src/http/axios.js +0 -372
  172. package/src/http/mock.interceptor.ts +0 -9
  173. package/src/http/obfuscationKey.ts +0 -41
  174. package/src/i18n.ts +0 -60
  175. package/src/index.js +0 -1
  176. package/src/index.less +0 -169
  177. package/src/locales/en/auth.ts +0 -70
  178. package/src/locales/en/base/memory.ts +0 -28
  179. package/src/locales/en/base/settings.ts +0 -41
  180. package/src/locales/en/chat.ts +0 -40
  181. package/src/locales/en/common.ts +0 -173
  182. package/src/locales/en/enum.ts +0 -27
  183. package/src/locales/en/menus/business-rules.ts +0 -48
  184. package/src/locales/en/menus/feedback.ts +0 -62
  185. package/src/locales/en/menus/knowledge.ts +0 -120
  186. package/src/locales/en/menus/model-metadata/index.ts +0 -10
  187. package/src/locales/en/menus/model-metadata/manage.ts +0 -151
  188. package/src/locales/en/menus/model-metadata/role.ts +0 -48
  189. package/src/locales/en/menus/model-metadata/training.ts +0 -65
  190. package/src/locales/en/menus/skill.ts +0 -34
  191. package/src/locales/en/system/agent-config.ts +0 -34
  192. package/src/locales/en/system/department.ts +0 -68
  193. package/src/locales/en/system/dict.ts +0 -44
  194. package/src/locales/en/system/menu.ts +0 -45
  195. package/src/locales/en/system/permission.ts +0 -89
  196. package/src/locales/en/system/role.ts +0 -25
  197. package/src/locales/en/system/sys-config.ts +0 -33
  198. package/src/locales/en/system/sys-log.ts +0 -38
  199. package/src/locales/en/system/user.ts +0 -113
  200. package/src/locales/en.ts +0 -68
  201. package/src/locales/zh/auth.ts +0 -70
  202. package/src/locales/zh/base/memory.ts +0 -29
  203. package/src/locales/zh/base/settings.ts +0 -41
  204. package/src/locales/zh/chat.ts +0 -47
  205. package/src/locales/zh/common.ts +0 -178
  206. package/src/locales/zh/enum.ts +0 -28
  207. package/src/locales/zh/menus/business-rules.ts +0 -47
  208. package/src/locales/zh/menus/feedback.ts +0 -62
  209. package/src/locales/zh/menus/knowledge.ts +0 -117
  210. package/src/locales/zh/menus/model-metadata/index.ts +0 -10
  211. package/src/locales/zh/menus/model-metadata/manage.ts +0 -151
  212. package/src/locales/zh/menus/model-metadata/role.ts +0 -47
  213. package/src/locales/zh/menus/model-metadata/training.ts +0 -64
  214. package/src/locales/zh/menus/skill.ts +0 -34
  215. package/src/locales/zh/system/agent-config.ts +0 -33
  216. package/src/locales/zh/system/department.ts +0 -69
  217. package/src/locales/zh/system/dict.ts +0 -44
  218. package/src/locales/zh/system/menu.ts +0 -47
  219. package/src/locales/zh/system/permission.ts +0 -94
  220. package/src/locales/zh/system/role.ts +0 -25
  221. package/src/locales/zh/system/sys-config.ts +0 -32
  222. package/src/locales/zh/system/sys-log.ts +0 -38
  223. package/src/locales/zh/system/user.ts +0 -114
  224. package/src/locales/zh.ts +0 -67
  225. package/src/main.tsx +0 -50
  226. package/src/meta/const/index.ts +0 -40
  227. package/src/meta/index-dict.ts +0 -56
  228. package/src/meta/index-enum.ts +0 -95
  229. package/src/meta/index.ts +0 -14
  230. package/src/meta/module/dict-data/runtime.ts +0 -199
  231. package/src/meta/module/dict-data/types.ts +0 -17
  232. package/src/meta/module/enum-data/runtime.ts +0 -75
  233. package/src/meta/module/enum-data/types.ts +0 -18
  234. package/src/router/index.tsx +0 -312
  235. package/src/styles/AntdThemeProvider.tsx +0 -40
  236. package/src/styles/antd-theme.ts +0 -20
  237. package/src/styles/global.less +0 -107
  238. package/src/styles/variable.less +0 -103
  239. package/src/types/feedback.ts +0 -43
  240. package/src/types/index.ts +0 -85
  241. package/src/types/menu.ts +0 -43
  242. package/src/utils/aesUtil.ts +0 -123
  243. package/src/utils/chatUtils.ts +0 -524
  244. package/src/utils/cn.ts +0 -6
  245. package/src/utils/crypto.ts +0 -164
  246. package/src/utils/date.ts +0 -72
  247. package/src/utils/file-icons.tsx +0 -79
  248. package/src/utils/index.ts +0 -168
  249. package/src/utils/markdown-math-plugins.ts +0 -21
  250. package/src/utils/menuI18n.ts +0 -305
  251. package/src/utils/menuRouteRegistry.ts +0 -78
  252. package/src/utils/permission-crud.ts +0 -147
  253. package/src/utils/routeConfig.ts +0 -350
  254. package/src/utils/storage.ts +0 -135
  255. package/src/utils/toastBridge.ts +0 -26
  256. package/src/utils/url.ts +0 -38
  257. package/src/utils/validation.ts +0 -16
  258. package/src/views/auth/auth-code/index.less +0 -169
  259. package/src/views/auth/auth-code/index.module.less +0 -174
  260. package/src/views/auth/auth-code/index.tsx +0 -233
  261. package/src/views/auth/login.tsx +0 -498
  262. package/src/views/auth/register.tsx +0 -388
  263. package/src/views/base/memory/index.tsx +0 -136
  264. package/src/views/base/memory/modal/detail-modal.tsx +0 -89
  265. package/src/views/base/memory/modal/submit-modal.tsx +0 -134
  266. package/src/views/base/settings/index.tsx +0 -657
  267. package/src/views/chat/chat.less +0 -323
  268. package/src/views/chat/components/chat-input.tsx +0 -298
  269. package/src/views/chat/components/header-thread-title.tsx +0 -210
  270. package/src/views/chat/components/message-list/content-answer.tsx +0 -100
  271. package/src/views/chat/components/message-list/content-question.tsx +0 -18
  272. package/src/views/chat/components/message-list/index.tsx +0 -520
  273. package/src/views/chat/components/message-list/message-item.tsx +0 -199
  274. package/src/views/chat/components/message-list/preparation-demo-items.ts +0 -147
  275. package/src/views/chat/components/message-list/preparation-steps.tsx +0 -506
  276. package/src/views/chat/components/message-list/suggestion-list.tsx +0 -36
  277. package/src/views/chat/components/message-list/thinking-process.tsx +0 -49
  278. package/src/views/chat/components/message-list/toolbar.tsx +0 -224
  279. package/src/views/chat/components/message-list/use-message-list-scroll.ts +0 -214
  280. package/src/views/chat/components/references-knowledge/context.tsx +0 -57
  281. package/src/views/chat/components/references-knowledge/index.ts +0 -9
  282. package/src/views/chat/components/references-knowledge/modal/knowledge-detail-drawer.tsx +0 -556
  283. package/src/views/chat/components/references-knowledge/modal/knowledge-doc-detail-drawer.tsx +0 -529
  284. package/src/views/chat/components/references-knowledge/panel.tsx +0 -115
  285. package/src/views/chat/hooks/use-chat-common.ts +0 -19
  286. package/src/views/chat/index-session.tsx +0 -647
  287. package/src/views/chat/index.tsx +0 -127
  288. package/src/views/page-error/401.tsx +0 -56
  289. package/src/views/page-error/404.tsx +0 -56
  290. package/src/views/page-menus/business-rules/index.tsx +0 -376
  291. package/src/views/page-menus/business-rules/modal/detail-modal.tsx +0 -186
  292. package/src/views/page-menus/business-rules/modal/scope-modal.tsx +0 -272
  293. package/src/views/page-menus/business-rules/modal/submit-modal.tsx +0 -142
  294. package/src/views/page-menus/feedback/components/feedback-dataset-list.tsx +0 -471
  295. package/src/views/page-menus/feedback/index.tsx +0 -166
  296. package/src/views/page-menus/feedback/modal/export-feedback-modal.tsx +0 -367
  297. package/src/views/page-menus/knowledge/components/doc-editor-by-type.tsx +0 -32
  298. package/src/views/page-menus/knowledge/components/doc-editor-type-file.tsx +0 -330
  299. package/src/views/page-menus/knowledge/detail.tsx +0 -600
  300. package/src/views/page-menus/knowledge/index.tsx +0 -337
  301. package/src/views/page-menus/knowledge/modal/detail-modal.tsx +0 -618
  302. package/src/views/page-menus/knowledge/modal/doc-detail-modal.tsx +0 -550
  303. package/src/views/page-menus/knowledge/modal/doc-parse.ts +0 -99
  304. package/src/views/page-menus/knowledge/modal/doc-submit-modal.tsx +0 -349
  305. package/src/views/page-menus/knowledge/modal/doc-type-picker-modal.tsx +0 -88
  306. package/src/views/page-menus/knowledge/modal/knowledge-user-select-modal.tsx +0 -283
  307. package/src/views/page-menus/knowledge/modal/submit-modal.tsx +0 -179
  308. package/src/views/page-menus/model-metadata/manage/components/metadata-detail-schema-tab.tsx +0 -114
  309. package/src/views/page-menus/model-metadata/manage/components/step1-basic-info.tsx +0 -232
  310. package/src/views/page-menus/model-metadata/manage/components/step2-schema.tsx +0 -316
  311. package/src/views/page-menus/model-metadata/manage/components/step3-permissions.tsx +0 -134
  312. package/src/views/page-menus/model-metadata/manage/components/step4-documents.tsx +0 -134
  313. package/src/views/page-menus/model-metadata/manage/components/step5-example-sql.tsx +0 -101
  314. package/src/views/page-menus/model-metadata/manage/components/submit-add.tsx +0 -338
  315. package/src/views/page-menus/model-metadata/manage/components/submit-edit.tsx +0 -276
  316. package/src/views/page-menus/model-metadata/manage/detail.tsx +0 -298
  317. package/src/views/page-menus/model-metadata/manage/hooks/model-metadata-submit-shared.ts +0 -113
  318. package/src/views/page-menus/model-metadata/manage/hooks/use-model-metadata-item-state.ts +0 -20
  319. package/src/views/page-menus/model-metadata/manage/index.tsx +0 -304
  320. package/src/views/page-menus/model-metadata/manage/modal/components/table-schema.ts +0 -374
  321. package/src/views/page-menus/model-metadata/manage/modal/components/use-table-detail-tabs.tsx +0 -151
  322. package/src/views/page-menus/model-metadata/manage/modal/components/use-table-submit-tabs.tsx +0 -423
  323. package/src/views/page-menus/model-metadata/manage/modal/detail-modal.tsx +0 -218
  324. package/src/views/page-menus/model-metadata/manage/modal/submit-modal.tsx +0 -261
  325. package/src/views/page-menus/model-metadata/manage/modal/table-detail-modal.tsx +0 -196
  326. package/src/views/page-menus/model-metadata/manage/modal/table-submit-modal.tsx +0 -229
  327. package/src/views/page-menus/model-metadata/manage/submit.tsx +0 -31
  328. package/src/views/page-menus/model-metadata/role/index.tsx +0 -207
  329. package/src/views/page-menus/model-metadata/role/modal/detail-modal.tsx +0 -97
  330. package/src/views/page-menus/model-metadata/role/modal/role-assign-users-modal.tsx +0 -254
  331. package/src/views/page-menus/model-metadata/role/modal/role-assign-users-panel.tsx +0 -393
  332. package/src/views/page-menus/model-metadata/role/modal/role-assign-users-utils.ts +0 -120
  333. package/src/views/page-menus/model-metadata/role/modal/role-permission-assign-panel.tsx +0 -698
  334. package/src/views/page-menus/model-metadata/role/modal/role-permission-modal.tsx +0 -237
  335. package/src/views/page-menus/model-metadata/role/modal/submit-modal.tsx +0 -135
  336. package/src/views/page-menus/model-metadata/training/components/detail-records/index.ts +0 -4
  337. package/src/views/page-menus/model-metadata/training/components/detail-records/node-card.tsx +0 -72
  338. package/src/views/page-menus/model-metadata/training/components/detail-records/summary-lines.ts +0 -196
  339. package/src/views/page-menus/model-metadata/training/components/detail-records/summary-list.tsx +0 -153
  340. package/src/views/page-menus/model-metadata/training/components/detail-records/timeline.tsx +0 -103
  341. package/src/views/page-menus/model-metadata/training/components/detail-records/types.ts +0 -82
  342. package/src/views/page-menus/model-metadata/training/detail.tsx +0 -159
  343. package/src/views/page-menus/model-metadata/training/index.tsx +0 -236
  344. package/src/views/page-menus/model-metadata/training/modal/update-detail-modal.tsx +0 -154
  345. package/src/views/page-menus/skill/index.tsx +0 -201
  346. package/src/views/page-menus/skill/modal/detail-modal.tsx +0 -156
  347. package/src/views/page-menus/skill/modal/submit-modal.tsx +0 -214
  348. package/src/views/page-system/agent-config/index.tsx +0 -370
  349. package/src/views/page-system/department/departmentFormShared.ts +0 -36
  350. package/src/views/page-system/department/index.tsx +0 -541
  351. package/src/views/page-system/department/modal/detail-modal.tsx +0 -94
  352. package/src/views/page-system/department/modal/member-role-modal.tsx +0 -128
  353. package/src/views/page-system/department/modal/submit-modal.tsx +0 -265
  354. package/src/views/page-system/dict/index.tsx +0 -440
  355. package/src/views/page-system/dict/modal/cate-submit-modal.tsx +0 -315
  356. package/src/views/page-system/dict/modal/submit-modal.tsx +0 -184
  357. package/src/views/page-system/logs/components/index.ts +0 -3
  358. package/src/views/page-system/logs/components/log-message-demo.tsx +0 -30
  359. package/src/views/page-system/logs/components/log-message-stream.ts +0 -184
  360. package/src/views/page-system/logs/components/message-list/content-answer.tsx +0 -100
  361. package/src/views/page-system/logs/components/message-list/content-question.tsx +0 -18
  362. package/src/views/page-system/logs/components/message-list/index.tsx +0 -515
  363. package/src/views/page-system/logs/components/message-list/message-item.tsx +0 -193
  364. package/src/views/page-system/logs/components/message-list/preparation-demo-items.ts +0 -147
  365. package/src/views/page-system/logs/components/message-list/preparation-steps.tsx +0 -506
  366. package/src/views/page-system/logs/components/message-list/suggestion-list.tsx +0 -36
  367. package/src/views/page-system/logs/components/message-list/thinking-process.tsx +0 -49
  368. package/src/views/page-system/logs/components/message-list/toolbar.tsx +0 -134
  369. package/src/views/page-system/logs/components/message-list/use-message-list-scroll.ts +0 -214
  370. package/src/views/page-system/logs/components/message-modal.tsx +0 -239
  371. package/src/views/page-system/logs/index.tsx +0 -132
  372. package/src/views/page-system/logs/modal/detail-modal.tsx +0 -157
  373. package/src/views/page-system/menu/components/menuFormShared.ts +0 -283
  374. package/src/views/page-system/menu/index.less +0 -12
  375. package/src/views/page-system/menu/index.tsx +0 -410
  376. package/src/views/page-system/menu/modal/icon-modal.less +0 -51
  377. package/src/views/page-system/menu/modal/icon-modal.tsx +0 -87
  378. package/src/views/page-system/menu/modal/submit-modal.tsx +0 -263
  379. package/src/views/page-system/permission/index.tsx +0 -562
  380. package/src/views/page-system/permission/modal/detail-modal.tsx +0 -179
  381. package/src/views/page-system/permission/modal/submit-modal.less +0 -146
  382. package/src/views/page-system/permission/modal/submit-modal.tsx +0 -650
  383. package/src/views/page-system/role/index.tsx +0 -163
  384. package/src/views/page-system/role/modal/detail-modal.tsx +0 -127
  385. package/src/views/page-system/role/modal/permission-assign-group-rules.ts +0 -86
  386. package/src/views/page-system/role/modal/permission-modal.tsx +0 -111
  387. package/src/views/page-system/role/modal/role-modal-shell-styles.ts +0 -21
  388. package/src/views/page-system/role/modal/role-permission-assign-panel.tsx +0 -916
  389. package/src/views/page-system/role/modal/role-permission-assign-shared.ts +0 -1047
  390. package/src/views/page-system/role/modal/submit-modal.tsx +0 -193
  391. package/src/views/page-system/sys-config/index.tsx +0 -294
  392. package/src/views/page-system/user/components/user-role-column.tsx +0 -87
  393. package/src/views/page-system/user/index.tsx +0 -439
  394. package/src/views/page-system/user/modal/assign-roles-modal.tsx +0 -389
  395. package/src/views/page-system/user/modal/detail-modal.tsx +0 -72
  396. package/src/views/page-system/user/modal/modal-style/submit-modal.less +0 -40
  397. package/src/views/page-system/user/modal/submit-modal.less +0 -40
  398. package/src/views/page-system/user/modal/submit-modal.tsx +0 -287
  399. package/src/views/page-system/user/userFormShared.ts +0 -51
  400. package/tailwind.config.js +0 -17
  401. package/tsconfig.app.json +0 -48
  402. package/tsconfig.json +0 -11
  403. package/tsconfig.node.json +0 -26
  404. package/vite.config.ts +0 -272
  405. /package/{public → dist}/favicon.ico +0 -0
  406. /package/{public → dist}/vite.svg +0 -0
@@ -1,717 +0,0 @@
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
-