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,193 @@
1
+ import { addRoleAPI, editRoleAPI, getRoleDetailAPI } from '@/api/system/role'
2
+ import { useToast } from '@/contexts/ToastContext'
3
+ import { useModalSubmit } from '@/hooks/useModalSubmit'
4
+ import { CxForm, CxFormItem, CxInput, CxModal, useCxModal } from '@cx-ui'
5
+ import { Form } from 'antd'
6
+ import {
7
+ forwardRef,
8
+ useCallback,
9
+ useImperativeHandle,
10
+ useMemo,
11
+ useRef,
12
+ useState,
13
+ } from 'react'
14
+ import { useTranslation } from 'react-i18next'
15
+ import { roleModalShellStyles } from './role-modal-shell-styles'
16
+ import RolePermissionAssignPanel, {
17
+ type RolePermissionAssignPanelHandle,
18
+ } from './role-permission-assign-panel'
19
+ import {
20
+ EMPTY_ROLE_PERMISSION_CREATE_DTO,
21
+ type RolePermissionAssignPayload,
22
+ } from './role-permission-assign-shared'
23
+
24
+ interface RoleModalProps {
25
+ onOk?: (values: any) => void | Promise<void>
26
+ onCancel?: () => void
27
+ }
28
+
29
+ type ModalMode = 'add' | 'edit'
30
+ type OpenPayload = {
31
+ opeType?: ModalMode
32
+ opeId?: number | string
33
+ record?: any
34
+ }
35
+
36
+ function extractCreatedRoleId(result: unknown): string | number | undefined {
37
+ if (result == null || typeof result !== 'object') return undefined
38
+ const r = result as Record<string, unknown>
39
+ if (r.id != null) return r.id as string | number
40
+ const d = r.data
41
+ if (d && typeof d === 'object' && d !== null && 'id' in d) {
42
+ return (d as { id: string | number }).id
43
+ }
44
+ return undefined
45
+ }
46
+
47
+ const SubmitModal = forwardRef<any, RoleModalProps>(function SubmitModal(_props, ref) {
48
+ const { t } = useTranslation()
49
+ const { showToast } = useToast()
50
+ const { onOk, onCancel } = _props
51
+
52
+ const [assignBootstrapKey, setAssignBootstrapKey] = useState(0)
53
+ const assignPanelRef = useRef<RolePermissionAssignPanelHandle>(null)
54
+ const { submitting, handleSubmit } = useModalSubmit()
55
+
56
+ const modal = useCxModal({
57
+ onOpen: async (payload) => {
58
+ const nextType = payload.opeType ?? 'add'
59
+ const record = payload.data
60
+ const { form } = modal
61
+ if (nextType === 'add' || nextType === 'edit') {
62
+ setAssignBootstrapKey((k) => k + 1)
63
+ }
64
+ assignPanelRef.current?.reset()
65
+
66
+ if (nextType === 'edit' && record) {
67
+ const opeId = payload.opeId ?? record.id
68
+ if (opeId != null && `${opeId}`.trim() !== '') {
69
+ const detail = await getRoleDetailAPI(opeId)
70
+ form.setFieldsValue({
71
+ name: detail?.data?.name ?? '',
72
+ description: detail?.data?.description ?? '',
73
+ })
74
+ } else {
75
+ form.resetFields()
76
+ }
77
+ } else {
78
+ form.resetFields()
79
+ }
80
+ },
81
+ onClose: () => {
82
+ assignPanelRef.current?.reset()
83
+ onCancel?.()
84
+ },
85
+ })
86
+
87
+ const { isOpen, close, form, opeType, opeId } = modal
88
+
89
+ const title = useMemo(
90
+ () => (opeType === 'edit' ? t('system.role.edit') : t('system.role.add')),
91
+ [opeType, t],
92
+ )
93
+
94
+ const onSubmit = async () => {
95
+ if (opeType === 'detail') return
96
+ const result = await handleSubmit({
97
+ opeType,
98
+ validate: () => form.validateFields(),
99
+ addAPI: addRoleAPI,
100
+ editAPI: editRoleAPI,
101
+ transform: (values) => {
102
+ const p = assignPanelRef.current?.getPayload() as
103
+ | RolePermissionAssignPayload
104
+ | undefined
105
+ const rolePermissionCreateDTO =
106
+ p?.rolePermissionCreateDTO ?? EMPTY_ROLE_PERMISSION_CREATE_DTO
107
+ return {
108
+ name: values.name,
109
+ description: values.description ?? '',
110
+ rolePermissionCreateDTO,
111
+ ...(opeType === 'edit' && opeId != null ? { id: opeId } : {}),
112
+ }
113
+ },
114
+ onSuccess: async ({ payload }) => {
115
+ await onOk?.(payload)
116
+ },
117
+ })
118
+ if (result) close()
119
+ }
120
+
121
+ useImperativeHandle(ref, () => ({
122
+ open: modal.open,
123
+ close,
124
+ }), [modal.open, close])
125
+
126
+ /** 新增 / 编辑统一宽屏承载权限分配区 */
127
+ /** 宽屏便于权限卡片一行 3 列(与面板内 auto-fill 网格配合) */
128
+ const modalWidth = 1104
129
+
130
+ return (
131
+ <CxModal
132
+ title={title}
133
+ isOpen={isOpen}
134
+ onClose={close}
135
+ opeType={opeType}
136
+ submitting={submitting}
137
+ onSubmit={onSubmit}
138
+ modalConfig={{
139
+ width: modalWidth,
140
+ styles: roleModalShellStyles,
141
+ rootClassName: 'cx-modal-role-assign',
142
+ }}>
143
+ <CxForm
144
+ form={form}
145
+ layout="vertical"
146
+ className="flex h-full min-h-0 flex-1 flex-col overflow-hidden">
147
+ <div className="grid shrink-0 grid-cols-1 gap-4 md:grid-cols-2">
148
+ <CxFormItem
149
+ label={t('system.role.name')}
150
+ name="name"
151
+ rules={[
152
+ {
153
+ required: true,
154
+ message: t('system.role.name_required'),
155
+ },
156
+ ]}>
157
+ <CxInput.Large placeholder={t('system.role.name')} />
158
+ </CxFormItem>
159
+
160
+ <CxFormItem
161
+ label={t('common.description')}
162
+ name="description"
163
+ rules={[
164
+ {
165
+ required: true,
166
+ message: t('common.description_required'),
167
+ },
168
+ ]}>
169
+ <CxInput.Large placeholder={t('common.description_placeholder')} />
170
+ </CxFormItem>
171
+ </div>
172
+ <div className="flex min-h-0 flex-1 flex-col gap-2 overflow-hidden">
173
+ <h3 className="shrink-0 text-sm font-semibold text-gray-900 dark:text-gray-100">
174
+ {t('system.role.permissions_assignment')}
175
+ </h3>
176
+ <div className="flex min-h-0 flex-1 flex-col overflow-hidden">
177
+ <RolePermissionAssignPanel
178
+ ref={assignPanelRef}
179
+ active={isOpen}
180
+ bootstrapKey={assignBootstrapKey}
181
+ maxHeightClass="h-full min-h-0 w-full flex-1 overflow-hidden"
182
+ roleId={opeType === 'edit' ? (opeId ?? null) : null}
183
+ />
184
+ </div>
185
+ </div>
186
+ </CxForm>
187
+ </CxModal>
188
+ )
189
+ })
190
+
191
+ SubmitModal.displayName = 'SubmitModal'
192
+
193
+ export default SubmitModal
@@ -0,0 +1,294 @@
1
+ import { Form, Switch } from 'antd'
2
+ import { RefreshCw, Save, Settings as SettingsIcon } from 'lucide-react'
3
+ import React, { useEffect, useMemo, useRef, useState } from 'react'
4
+ import { useTranslation } from 'react-i18next'
5
+
6
+ import { getSysConfigAPI, saveSysConfigAPI } from '@/api/system/sys-config'
7
+ import { loadSystemConfigOnBoot } from '@/config/system'
8
+ import { CxButton, CxForm, CxFormItem, CxInput, UploadImg } from '@cx-ui'
9
+ import { useToast } from '../../../contexts/ToastContext'
10
+ import { useStickyHeader } from '../../../hooks/useStickyHeader'
11
+ type SysConfigForm = {
12
+ systemName: string
13
+ copyright: string
14
+ systemLogo: string[] // 上传返回的 address 数组
15
+ loginPageName: string
16
+ browserIcon: string[] // 上传返回的 address 数组
17
+ showCopyright: boolean
18
+ agentAvatar: string[] // 上传返回的 address 数组
19
+ }
20
+
21
+ const SYS_CONFIG_KEYS = {
22
+ systemName: 'SYSTEM_NAME',
23
+ copyright: 'SYSTEM_COPYRIGHT',
24
+ systemLogo: 'SYSTEM_LOGO',
25
+ loginPageName: 'LOGIN_PAGE_NAME',
26
+ browserIcon: 'BROWSER_ICON',
27
+ showCopyright: 'SHOW_COPYRIGHT',
28
+ agentAvatar: 'AGENT_AVATAR',
29
+ } as const
30
+
31
+ const SysConfigPage: React.FC = () => {
32
+ const { t } = useTranslation()
33
+ const { showToast } = useToast()
34
+ const titleRef = useStickyHeader(t('system.sys_config.title'))
35
+
36
+ const [isLoading, setIsLoading] = useState(true)
37
+ const [isSaving, setIsSaving] = useState(false)
38
+ const [formTick, setFormTick] = useState(0)
39
+ const hasFetchedOnceRef = useRef(false)
40
+
41
+ const [form] = Form.useForm<SysConfigForm>()
42
+ const browserIconValue = Form.useWatch('browserIcon', form)
43
+ const systemLogoValue = Form.useWatch('systemLogo', form)
44
+ const agentAvatarValue = Form.useWatch('agentAvatar', form)
45
+ const [original, setOriginal] = useState<SysConfigForm>({
46
+ systemName: '',
47
+ copyright: '',
48
+ systemLogo: [],
49
+ loginPageName: '',
50
+ browserIcon: [],
51
+ showCopyright: true,
52
+ agentAvatar: [],
53
+ })
54
+
55
+ const isModified = useMemo(() => {
56
+ const cur = (form.getFieldsValue(true) || {}) as SysConfigForm
57
+ return (
58
+ (cur.systemName ?? '') !== (original.systemName ?? '') ||
59
+ (cur.copyright ?? '') !== (original.copyright ?? '') ||
60
+ (cur.systemLogo?.[0] ?? '') !== (original.systemLogo?.[0] ?? '') ||
61
+ (cur.loginPageName ?? '') !== (original.loginPageName ?? '') ||
62
+ (cur.browserIcon?.[0] ?? '') !== (original.browserIcon?.[0] ?? '') ||
63
+ Boolean(cur.showCopyright) !== Boolean(original.showCopyright) ||
64
+ (cur.agentAvatar?.[0] ?? '') !== (original.agentAvatar?.[0] ?? '')
65
+ )
66
+ }, [form, original, formTick])
67
+
68
+ /** 将表单值转换为后端需要的 configs 对象。 */
69
+ const buildConfigsObject = (values: SysConfigForm) => {
70
+ return {
71
+ [SYS_CONFIG_KEYS.systemName]: values.systemName ?? '',
72
+ [SYS_CONFIG_KEYS.copyright]: values.copyright ?? '',
73
+ [SYS_CONFIG_KEYS.systemLogo]: values.systemLogo?.[0] ?? '',
74
+ [SYS_CONFIG_KEYS.loginPageName]: values.loginPageName ?? '',
75
+ [SYS_CONFIG_KEYS.browserIcon]: values.browserIcon?.[0] ?? '',
76
+ [SYS_CONFIG_KEYS.showCopyright]: values.showCopyright ? 1 : 0,
77
+ [SYS_CONFIG_KEYS.agentAvatar]: values.agentAvatar?.[0] ?? '',
78
+ }
79
+ }
80
+
81
+ /** 将后端返回的 configs 对象转换为表单需要的字段。 */
82
+ const buildFormValues = (configs: Record<string, any>): SysConfigForm => {
83
+ const rawShow = configs[SYS_CONFIG_KEYS.showCopyright]
84
+ const showCopyright =
85
+ rawShow === undefined || rawShow === null || rawShow === ''
86
+ ? true
87
+ : rawShow === 1 || rawShow === '1' || rawShow === true || rawShow === 'true'
88
+ return {
89
+ systemName: configs[SYS_CONFIG_KEYS.systemName] ?? '',
90
+ copyright: configs[SYS_CONFIG_KEYS.copyright] ?? '',
91
+ systemLogo: configs[SYS_CONFIG_KEYS.systemLogo] ? [String(configs[SYS_CONFIG_KEYS.systemLogo])] : [],
92
+ loginPageName: configs[SYS_CONFIG_KEYS.loginPageName] ?? '',
93
+ browserIcon: configs[SYS_CONFIG_KEYS.browserIcon] ? [String(configs[SYS_CONFIG_KEYS.browserIcon])] : [],
94
+ showCopyright,
95
+ agentAvatar: configs[SYS_CONFIG_KEYS.agentAvatar] ? [String(configs[SYS_CONFIG_KEYS.agentAvatar])] : [],
96
+ }
97
+ }
98
+
99
+ /** 拉取系统配置并填充表单。 */
100
+ const fetchConfig = async () => {
101
+ setIsLoading(true)
102
+ try {
103
+ const res: any = await getSysConfigAPI({ type: 'pageConfig' })
104
+ // 后台返回详情是 res.data,且为字符串对象(JSON string)
105
+ const configsText = res.data
106
+ const configsObj = JSON.parse(configsText || '{}')
107
+ const next = buildFormValues(configsObj)
108
+ form.setFieldsValue(next)
109
+ setOriginal(next)
110
+ } catch (e) {
111
+ console.error('Failed to fetch sys config:', e)
112
+ } finally {
113
+ setIsLoading(false)
114
+ }
115
+ }
116
+
117
+ useEffect(() => {
118
+ // React StrictMode(开发环境)下 useEffect 会触发两次,这里只在首次挂载时拉取一次
119
+ if (hasFetchedOnceRef.current) return
120
+ hasFetchedOnceRef.current = true
121
+ void fetchConfig()
122
+ }, [])
123
+
124
+ /** 保存系统配置:type 固定为 pageConfig,configs 为字符串对象(JSON string)。 */
125
+ const handleSave = async () => {
126
+ if (!isModified) return
127
+ setIsSaving(true)
128
+ try {
129
+ const cur = (form.getFieldsValue(true) || {}) as SysConfigForm
130
+ const configsObj = buildConfigsObject(cur)
131
+ await saveSysConfigAPI({
132
+ type: 'pageConfig',
133
+ configs: JSON.stringify(configsObj),
134
+ })
135
+ setOriginal(cur)
136
+ showToast(t('system.sys_config.messages.save_success'), 'success')
137
+ // 保存后立即重新应用系统配置(标题/图标/会话头像等),避免用户手动刷新
138
+ await loadSystemConfigOnBoot()
139
+ } catch (e) {
140
+ console.error('Failed to save sys config:', e)
141
+ } finally {
142
+ setIsSaving(false)
143
+ }
144
+ }
145
+
146
+ /** 上传系统 Logo 成功后回填表单字段(地址数组)。 */
147
+ const handleSystemLogoUploaded = (fileList: any[]) => {
148
+ const addresses = (Array.isArray(fileList) ? fileList : [])
149
+ .map((f) => f?.response?.data?.address ?? f?.address ?? f?.url ?? '')
150
+ .filter(Boolean)
151
+ .map((x) => String(x))
152
+ form.setFieldValue('systemLogo', addresses)
153
+ // form.setFieldValue 不触发表单 onValuesChange,这里手动触发一次 modified 计算刷新
154
+ setFormTick((x) => x + 1)
155
+ }
156
+
157
+ /** 上传浏览器图标成功后回填表单字段(地址数组)。 */
158
+ const handleBrowserIconUploaded = (fileList: any[]) => {
159
+ const addresses = (Array.isArray(fileList) ? fileList : [])
160
+ .map((f) => f?.response?.data?.address ?? f?.address ?? f?.url ?? '')
161
+ .filter(Boolean)
162
+ .map((x) => String(x))
163
+ form.setFieldValue('browserIcon', addresses)
164
+ // form.setFieldValue 不触发表单 onValuesChange,这里手动触发一次 modified 计算刷新
165
+ setFormTick((x) => x + 1)
166
+ }
167
+
168
+ /** 上传 Agent 头像成功后回填表单字段(地址数组)。 */
169
+ const handleAgentAvatarUploaded = (fileList: any[]) => {
170
+ const addresses = (Array.isArray(fileList) ? fileList : [])
171
+ .map((f) => f?.response?.data?.address ?? f?.address ?? f?.url ?? '')
172
+ .filter(Boolean)
173
+ .map((x) => String(x))
174
+ form.setFieldValue('agentAvatar', addresses)
175
+ // form.setFieldValue 不触发表单 onValuesChange,这里手动触发一次 modified 计算刷新
176
+ setFormTick((x) => x + 1)
177
+ }
178
+
179
+ if (isLoading) {
180
+ return (
181
+ <div className="flex items-center justify-center p-12">
182
+ <RefreshCw size={32} className="animate-spin text-blue-600" />
183
+ </div>
184
+ )
185
+ }
186
+
187
+ return (
188
+ <div className="p-8 mx-auto space-y-8 pb-20">
189
+ <div className="flex justify-between items-center" ref={titleRef}>
190
+ <div>
191
+ <h1 className="text-3xl font-bold bg-clip-text text-transparent bg-gradient-to-r from-blue-400 to-purple-500 flex items-center gap-3">
192
+ <SettingsIcon className="text-blue-400" size={32} />
193
+ {t('system.sys_config.title')}
194
+ </h1>
195
+ <p className="text-gray-500 dark:text-gray-400 mt-1">
196
+ {t('system.sys_config.description')}
197
+ </p>
198
+ </div>
199
+
200
+ <div className="flex items-center gap-2">
201
+ <CxButton.Icon
202
+ type="default"
203
+ icon={<RefreshCw size={20} />}
204
+ title={t('common.refresh')}
205
+ onClick={fetchConfig}
206
+ className="text-gray-500"
207
+ />
208
+ <CxButton.Icon
209
+ type="primary"
210
+ icon={isSaving ? <RefreshCw size={20} className="animate-spin" /> : <Save size={20} />}
211
+ title={t('common.save')}
212
+ disabled={!isModified || isSaving}
213
+ onClick={handleSave}
214
+ />
215
+ </div>
216
+ </div>
217
+
218
+ {/* Warning Banner (已按需求隐藏) */}
219
+
220
+ {/* 单表单区域 */}
221
+ <div className="bg-white dark:bg-gray-800 rounded-xl shadow-sm border border-gray-200 dark:border-gray-700 overflow-hidden p-6">
222
+ <section className="space-y-6">
223
+ <h2 className="text-lg font-semibold flex items-center gap-2 text-gray-900 dark:text-white border-b pb-4 dark:border-gray-700">
224
+ <SettingsIcon size={20} className="text-blue-500" />
225
+ {t('system.sys_config.form.title')}
226
+ </h2>
227
+
228
+ <CxForm
229
+ form={form}
230
+ layout="horizontal"
231
+ labelCol={{ style: { width: 140 } }}
232
+ wrapperCol={{ style: { flex: 1 } }}
233
+ onValuesChange={() => setFormTick((x) => x + 1)}
234
+ >
235
+ <CxFormItem label={t('system.sys_config.form.system_name')} name="systemName">
236
+ <CxInput.Large
237
+ style={{ width: 400 }}
238
+ placeholder={t('system.sys_config.form.system_name_placeholder')}
239
+ />
240
+ </CxFormItem>
241
+
242
+ <CxFormItem label={t('system.sys_config.form.copyright')} name="copyright">
243
+ <CxInput.Large
244
+ style={{ width: 400 }}
245
+ placeholder={t('system.sys_config.form.copyright_placeholder')}
246
+ />
247
+ </CxFormItem>
248
+
249
+ <CxFormItem label={t('system.sys_config.form.browser_title')} name="loginPageName">
250
+ <CxInput.Large
251
+ style={{ width: 400 }}
252
+ placeholder={t('system.sys_config.form.browser_title_placeholder')}
253
+ />
254
+ </CxFormItem>
255
+
256
+ <CxFormItem
257
+ label={t('system.sys_config.form.show_copyright')}
258
+ name="showCopyright"
259
+ valuePropName="checked"
260
+ >
261
+ <Switch />
262
+ </CxFormItem>
263
+
264
+ <CxFormItem label={t('system.sys_config.form.system_logo')} name="systemLogo">
265
+ <UploadImg
266
+ isPublic
267
+ value={Array.isArray(systemLogoValue) ? systemLogoValue : []}
268
+ onUploaded={handleSystemLogoUploaded}
269
+ />
270
+ </CxFormItem>
271
+
272
+ <CxFormItem label={t('system.sys_config.form.browser_icon')} name="browserIcon">
273
+ <UploadImg
274
+ isPublic
275
+ value={Array.isArray(browserIconValue) ? browserIconValue : []}
276
+ onUploaded={handleBrowserIconUploaded}
277
+ />
278
+ </CxFormItem>
279
+
280
+ <CxFormItem label={t('system.sys_config.form.agent_avatar')} name="agentAvatar">
281
+ <UploadImg
282
+ isPublic
283
+ value={Array.isArray(agentAvatarValue) ? agentAvatarValue : []}
284
+ onUploaded={handleAgentAvatarUploaded}
285
+ />
286
+ </CxFormItem>
287
+ </CxForm>
288
+ </section>
289
+ </div>
290
+ </div>
291
+ )
292
+ }
293
+
294
+ export default SysConfigPage
@@ -0,0 +1,87 @@
1
+ import { CxTooltip } from '@cx-ui'
2
+ import clsx from 'clsx'
3
+
4
+ const MAX_VISIBLE_ROLES = 3
5
+
6
+ function parseRoleList(roleNames: any, record: any): string[] {
7
+ const raw = roleNames ?? record?.roleNames ?? record?.roles ?? []
8
+ if (Array.isArray(raw)) {
9
+ return raw
10
+ .map((item) => {
11
+ if (typeof item === 'string') return item
12
+ if (item && typeof item === 'object') return item.name ?? item.roleName ?? item.label
13
+ return ''
14
+ })
15
+ .filter(Boolean)
16
+ }
17
+ if (typeof raw === 'string') {
18
+ return raw
19
+ .split(',')
20
+ .map((s) => s.trim())
21
+ .filter(Boolean)
22
+ }
23
+ return []
24
+ }
25
+
26
+ function RoleChip({ roleName }: { roleName: string }) {
27
+ const normalizedRoleName = String(roleName || '').toLowerCase()
28
+ const chipClassName =
29
+ normalizedRoleName === 'default'
30
+ ? 'bg-blue-100 text-blue-800 dark:bg-blue-900/30 dark:text-blue-300'
31
+ : normalizedRoleName === 'admin'
32
+ ? 'bg-purple-100 text-purple-800 dark:bg-purple-900/30 dark:text-purple-300'
33
+ : 'bg-gray-100 text-gray-700 dark:bg-gray-800/60 dark:text-gray-200'
34
+ return (
35
+ <span
36
+ className={clsx(
37
+ 'inline-flex items-center px-2 py-0.5 rounded text-xs font-medium whitespace-nowrap',
38
+ chipClassName,
39
+ )}>
40
+ {roleName}
41
+ </span>
42
+ )
43
+ }
44
+
45
+ export type UserRoleColumnProps = {
46
+ roleNames?: any
47
+ record?: any
48
+ }
49
+
50
+ export function UserRoleColumn({ roleNames, record }: UserRoleColumnProps) {
51
+ const roleList = parseRoleList(roleNames, record)
52
+
53
+ if (roleList.length === 0) return '-'
54
+
55
+ const visibleRoles = roleList.slice(0, MAX_VISIBLE_ROLES)
56
+ const hasOverflow = roleList.length > MAX_VISIBLE_ROLES
57
+
58
+ const compactView = (
59
+ <div className="flex flex-wrap items-center gap-1">
60
+ {visibleRoles.map((roleName, idx) => (
61
+ <RoleChip key={`${String(roleName)}-${idx}`} roleName={roleName} />
62
+ ))}
63
+ {hasOverflow && (
64
+ <span className="inline-flex items-center px-1 text-xs font-medium text-gray-500 dark:text-gray-400">
65
+
66
+ </span>
67
+ )}
68
+ </div>
69
+ )
70
+
71
+ if (!hasOverflow) {
72
+ return compactView
73
+ }
74
+
75
+ return (
76
+ <CxTooltip
77
+ title={
78
+ <div className="flex flex-wrap items-center justify-center gap-1">
79
+ {roleList.map((roleName, idx) => (
80
+ <RoleChip key={`${String(roleName)}-${idx}`} roleName={roleName} />
81
+ ))}
82
+ </div>
83
+ }>
84
+ <div className="inline-flex max-w-full cursor-default justify-center">{compactView}</div>
85
+ </CxTooltip>
86
+ )
87
+ }