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,254 @@
1
+ import DepUserSelectMultiModal, {
2
+ pickUserId,
3
+ type DepUserSelectUser,
4
+ } from '@/components/common/modal/dep-user-select-multi'
5
+ import { useToast } from '@/contexts/ToastContext'
6
+ import { roleModalShellStyles } from '@/views/page-system/role/modal/role-modal-shell-styles'
7
+ import {
8
+ getModelMetadataRoleAssignUsersAPI,
9
+ getModelMetadataRoleAssignUsersContextAPI,
10
+ saveModelMetadataRoleAssignUsersAPI,
11
+ } from '@api/menus/model-metadata/role'
12
+ import { CxModal, useCxModal } from '@cx-ui'
13
+ import { forwardRef, useCallback, useImperativeHandle, useRef, useState } from 'react'
14
+ import { useTranslation } from 'react-i18next'
15
+
16
+ import RoleAssignUsersPanel from './role-assign-users-panel'
17
+ import {
18
+ buildEmptyPermissionsFromContext,
19
+ mergeUserPermissionsWithContext,
20
+ parseLibrariesFromContext,
21
+ parseUsersFromAssignResponse,
22
+ type AssignContextLibrary,
23
+ type AssignUserEntry,
24
+ } from './role-assign-users-utils'
25
+
26
+ export type RoleAssignUsersModalOpenPayload = {
27
+ record: { id: string | number; name?: string }
28
+ }
29
+
30
+ const RoleAssignUsersModal = forwardRef<
31
+ { open: (p: RoleAssignUsersModalOpenPayload) => void; close: () => void },
32
+ { onSaved?: () => void | Promise<void> }
33
+ >(function RoleAssignUsersModal({ onSaved }, ref) {
34
+ const { t } = useTranslation()
35
+ const { showToast } = useToast()
36
+
37
+ const [loading, setLoading] = useState(false)
38
+ const [saving, setSaving] = useState(false)
39
+ const [roleId, setRoleId] = useState<string | number | null>(null)
40
+ const [titleHintFromList, setTitleHintFromList] = useState<string | undefined>(undefined)
41
+ const [libraries, setLibraries] = useState<AssignContextLibrary[]>([])
42
+ const [users, setUsers] = useState<AssignUserEntry[]>([])
43
+ const [activeUserId, setActiveUserId] = useState<string | null>(null)
44
+ const [panelBootstrapKey, setPanelBootstrapKey] = useState(0)
45
+
46
+ const librariesRef = useRef<AssignContextLibrary[]>([])
47
+ librariesRef.current = libraries
48
+
49
+ const userSelectRef = useRef<{ open: (p?: any) => void; close: () => void } | null>(null)
50
+
51
+ const modal = useCxModal({
52
+ onOpen: async (payload) => {
53
+ const id = payload?.data?.record?.id ?? payload?.opeId
54
+ setRoleId(id ?? null)
55
+ setLibraries([])
56
+ setUsers([])
57
+ setActiveUserId(null)
58
+ setPanelBootstrapKey((k) => k + 1)
59
+
60
+ if (id == null || `${id}`.trim() === '') {
61
+ setTitleHintFromList(undefined)
62
+ return
63
+ }
64
+
65
+ const hint = payload?.data?.record?.name
66
+ setTitleHintFromList(typeof hint === 'string' && hint.trim() ? hint.trim() : undefined)
67
+
68
+ setLoading(true)
69
+ try {
70
+ const [contextRes, assignRes] = await Promise.all([
71
+ getModelMetadataRoleAssignUsersContextAPI(id),
72
+ getModelMetadataRoleAssignUsersAPI(id),
73
+ ])
74
+ const libs = parseLibrariesFromContext(contextRes)
75
+ setLibraries(libs)
76
+ librariesRef.current = libs
77
+
78
+ const parsedUsers = parseUsersFromAssignResponse(assignRes).map((u) => ({
79
+ ...u,
80
+ permissions: mergeUserPermissionsWithContext(u.permissions, libs),
81
+ }))
82
+ setUsers(parsedUsers)
83
+ setActiveUserId(parsedUsers[0]?.userId ?? null)
84
+ } finally {
85
+ setLoading(false)
86
+ }
87
+ },
88
+ onClose: () => {
89
+ setRoleId(null)
90
+ setLibraries([])
91
+ setUsers([])
92
+ setActiveUserId(null)
93
+ setTitleHintFromList(undefined)
94
+ },
95
+ })
96
+
97
+ const { isOpen, close } = modal
98
+
99
+ const handleAddUsersClick = useCallback(() => {
100
+ userSelectRef.current?.open({
101
+ title: t('menus.model_metadata.role.assign_users_pick_users_title'), // 选择用户
102
+ deptSelectable: false,
103
+ })
104
+ }, [t])
105
+
106
+ const handleUserSelectOk = useCallback(
107
+ async ({ users: picked }: { users: DepUserSelectUser[] }) => {
108
+ const libs = librariesRef.current
109
+ let dupCount = 0
110
+ const newEntries: AssignUserEntry[] = []
111
+
112
+ setUsers((prev) => {
113
+ const existingIds = new Set(prev.map((u) => String(u.userId)))
114
+ for (const u of picked) {
115
+ const id = pickUserId(u)
116
+ if (id == null) continue
117
+ const sid = String(id)
118
+ if (existingIds.has(sid)) {
119
+ dupCount++
120
+ continue
121
+ }
122
+ existingIds.add(sid)
123
+ newEntries.push({
124
+ userId: sid,
125
+ userName:
126
+ u.display_name ?? u.userName ?? u.name ?? u.email ?? u.userEmail ?? sid,
127
+ userPhone: (u as any).userPhone ?? (u as any).phone ?? '',
128
+ permissions: buildEmptyPermissionsFromContext(libs),
129
+ })
130
+ }
131
+ if (!newEntries.length) return prev
132
+ return [...prev, ...newEntries]
133
+ })
134
+
135
+ if (newEntries.length > 0) {
136
+ setActiveUserId((cur) => cur ?? newEntries[0]?.userId ?? null)
137
+ }
138
+ if (dupCount > 0) {
139
+ showToast(t('menus.model_metadata.role.assign_users_duplicate'), 'info') // 已跳过重复用户
140
+ }
141
+ },
142
+ [showToast, t],
143
+ )
144
+
145
+ const handleRemoveUser = useCallback((userId: string) => {
146
+ setUsers((prev) => {
147
+ const next = prev.filter((u) => u.userId !== userId)
148
+ setActiveUserId((cur) => {
149
+ if (cur !== userId) return cur
150
+ return next[0]?.userId ?? null
151
+ })
152
+ return next
153
+ })
154
+ }, [])
155
+
156
+ const handlePermissionValueChange = useCallback(
157
+ (userId: string, metadataId: string, permissionId: string, value: string) => {
158
+ setUsers((prev) =>
159
+ prev.map((u) => {
160
+ if (u.userId !== userId) return u
161
+ return {
162
+ ...u,
163
+ permissions: u.permissions.map((p) =>
164
+ p.metadataId === metadataId && p.permissionId === permissionId
165
+ ? { ...p, permissionValue: value }
166
+ : p,
167
+ ),
168
+ }
169
+ }),
170
+ )
171
+ },
172
+ [],
173
+ )
174
+
175
+ const handleSave = useCallback(async () => {
176
+ if (!roleId) return
177
+ setSaving(true)
178
+ try {
179
+ await saveModelMetadataRoleAssignUsersAPI(roleId, {
180
+ users: users.map((u) => ({
181
+ userId: u.userId,
182
+ permissions: u.permissions.map((p) => ({
183
+ permissionValue: p.permissionValue ?? '',
184
+ metadataId: p.metadataId,
185
+ permissionId: p.permissionId,
186
+ })),
187
+ })),
188
+ })
189
+ showToast(t('common.operation_success'), 'success') // 操作成功
190
+ await onSaved?.()
191
+ close()
192
+ } catch {
193
+ // 失败由 axios 拦截器等统一提示
194
+ } finally {
195
+ setSaving(false)
196
+ }
197
+ }, [roleId, users, showToast, t, onSaved, close])
198
+
199
+ useImperativeHandle(
200
+ ref,
201
+ () => ({
202
+ open: (payload: RoleAssignUsersModalOpenPayload) => {
203
+ modal.open({
204
+ opeType: 'edit',
205
+ opeId: payload.record.id,
206
+ data: { record: payload.record },
207
+ })
208
+ },
209
+ close,
210
+ }),
211
+ [modal.open, close],
212
+ )
213
+
214
+ const titleBase = t('menus.model_metadata.role.assign_users_title') // 添加用户
215
+ const modalTitle = titleHintFromList ? `${titleBase} — ${titleHintFromList}` : titleBase
216
+
217
+ return (
218
+ <>
219
+ <CxModal
220
+ isOpen={isOpen}
221
+ title={modalTitle}
222
+ onClose={close}
223
+ opeType="edit"
224
+ onSubmit={() => void handleSave()}
225
+ submitting={saving}
226
+ modalConfig={{
227
+ width: 1380,
228
+ styles: roleModalShellStyles,
229
+ rootClassName: 'cx-modal-role-assign',
230
+ }}
231
+ >
232
+ <div className="flex h-full min-h-0 w-full flex-1 flex-col overflow-hidden">
233
+ <RoleAssignUsersPanel
234
+ key={`${String(roleId)}-${panelBootstrapKey}`}
235
+ active={isOpen}
236
+ loading={loading}
237
+ libraries={libraries}
238
+ users={users}
239
+ activeUserId={activeUserId}
240
+ onActiveUserIdChange={setActiveUserId}
241
+ onAddUsersClick={handleAddUsersClick}
242
+ onRemoveUser={handleRemoveUser}
243
+ onPermissionValueChange={handlePermissionValueChange}
244
+ maxHeightClass="h-full min-h-0 w-full flex-1 overflow-hidden"
245
+ />
246
+ </div>
247
+ </CxModal>
248
+ <DepUserSelectMultiModal ref={userSelectRef} onOk={handleUserSelectOk} />
249
+ </>
250
+ )
251
+ })
252
+
253
+ RoleAssignUsersModal.displayName = 'RoleAssignUsersModal'
254
+ export default RoleAssignUsersModal
@@ -0,0 +1,393 @@
1
+ // i18n 中文:locales/zh/menus/model-metadata/role.ts、manage.ts(permissionName / permissionColumns)
2
+
3
+ import { CxButton, CxInput, CxSkeleton, CxTable } from '@cx-ui'
4
+ import clsx from 'clsx'
5
+ import { Plus, Trash2 } from 'lucide-react'
6
+ import { useCallback, useEffect, useMemo, useRef, useState } from 'react'
7
+ import { useTranslation } from 'react-i18next'
8
+
9
+ import {
10
+ formatPermissionColumns,
11
+ type AssignContextLibrary,
12
+ type AssignUserEntry,
13
+ type UserPermissionRow,
14
+ } from './role-assign-users-utils'
15
+
16
+ type RoleAssignUsersPanelProps = {
17
+ active: boolean
18
+ loading: boolean
19
+ libraries: AssignContextLibrary[]
20
+ users: AssignUserEntry[]
21
+ activeUserId: string | null
22
+ onActiveUserIdChange: (userId: string) => void
23
+ onAddUsersClick: () => void
24
+ onRemoveUser: (userId: string) => void
25
+ onPermissionValueChange: (
26
+ userId: string,
27
+ metadataId: string,
28
+ permissionId: string,
29
+ value: string,
30
+ ) => void
31
+ maxHeightClass?: string
32
+ }
33
+
34
+ export default function RoleAssignUsersPanel(props: RoleAssignUsersPanelProps) {
35
+ const {
36
+ active,
37
+ loading,
38
+ libraries,
39
+ users,
40
+ activeUserId,
41
+ onActiveUserIdChange,
42
+ onAddUsersClick,
43
+ onRemoveUser,
44
+ onPermissionValueChange,
45
+ maxHeightClass,
46
+ } = props
47
+ const { t } = useTranslation()
48
+
49
+ const contentScrollRef = useRef<HTMLDivElement>(null)
50
+ const sectionRefs = useRef<Record<string, HTMLElement | null>>({})
51
+ const [activeMetadataId, setActiveMetadataId] = useState<string | null>(null)
52
+
53
+ const activeUser = useMemo(
54
+ () => users.find((u) => u.userId === activeUserId) ?? null,
55
+ [users, activeUserId],
56
+ )
57
+
58
+ const permRowsByMeta = useMemo(() => {
59
+ const map = new Map<string, UserPermissionRow[]>()
60
+ for (const row of activeUser?.permissions ?? []) {
61
+ const mid = row.metadataId
62
+ if (!map.has(mid)) map.set(mid, [])
63
+ map.get(mid)!.push(row)
64
+ }
65
+ return map
66
+ }, [activeUser?.permissions])
67
+
68
+ useEffect(() => {
69
+ if (!libraries.length) {
70
+ setActiveMetadataId(null)
71
+ return
72
+ }
73
+ setActiveMetadataId((cur) => {
74
+ if (cur && libraries.some((l) => l.metadataId === cur)) return cur
75
+ return libraries[0]?.metadataId ?? null
76
+ })
77
+ }, [libraries, activeUserId])
78
+
79
+ useEffect(() => {
80
+ if (loading || !activeUser || !libraries.length) return
81
+ const container = contentScrollRef.current
82
+ if (!container) return
83
+
84
+ const observer = new IntersectionObserver(
85
+ (entries) => {
86
+ const visible = entries
87
+ .filter((e) => e.isIntersecting)
88
+ .sort((a, b) => b.intersectionRatio - a.intersectionRatio)
89
+ const top = visible[0]
90
+ if (!top?.target) return
91
+ const id = top.target.getAttribute('data-metadata-id')
92
+ if (id) setActiveMetadataId(id)
93
+ },
94
+ { root: container, threshold: [0, 0.2, 0.45, 0.7], rootMargin: '-12% 0px -55% 0px' },
95
+ )
96
+
97
+ for (const lib of libraries) {
98
+ const el = sectionRefs.current[lib.metadataId]
99
+ if (el) {
100
+ el.setAttribute('data-metadata-id', lib.metadataId)
101
+ observer.observe(el)
102
+ }
103
+ }
104
+ return () => observer.disconnect()
105
+ }, [libraries, activeUser, loading])
106
+
107
+ const scrollToMeta = useCallback((metadataId: string) => {
108
+ setActiveMetadataId(metadataId)
109
+ const el = sectionRefs.current[metadataId]
110
+ const container = contentScrollRef.current
111
+ if (!el || !container) return
112
+ const top = el.offsetTop - container.offsetTop
113
+ container.scrollTo({ top: Math.max(0, top - 8), behavior: 'smooth' })
114
+ }, [])
115
+
116
+ const hasLayoutLock = Boolean(maxHeightClass && String(maxHeightClass).trim())
117
+ const rootLayoutClass = clsx(
118
+ 'flex min-h-0 gap-4 overflow-hidden',
119
+ hasLayoutLock ? clsx(maxHeightClass, 'min-h-0 flex-1') : 'min-h-[320px]',
120
+ )
121
+ const scrollColClass = 'min-h-0 flex-1 basis-0 overflow-y-auto overflow-x-hidden'
122
+ /** 与 role-permission-assign-panel 侧栏标题一致:固定 44px 高 */
123
+ const sidePanelHeaderClass =
124
+ 'flex h-[44px] min-h-[44px] shrink-0 items-center border-b border-gray-200 bg-gray-50 px-3 py-2.5 dark:border-gray-700 dark:bg-gray-700/50'
125
+ const sidePanelHeaderTitleClass =
126
+ 'line-clamp-1 min-w-0 flex-1 py-0.5 text-xs font-semibold uppercase tracking-wider text-gray-700 dark:text-gray-300'
127
+
128
+ if (!active) return null
129
+
130
+ const renderLibrarySection = (lib: AssignContextLibrary) => {
131
+ const mid = lib.metadataId
132
+ const templateMap = new Map(
133
+ (lib.permissions || []).map((p) => [p.permissionId, p] as const),
134
+ )
135
+ const rows = (permRowsByMeta.get(mid) ?? []).filter((r) => templateMap.has(r.permissionId))
136
+
137
+ const columns = [
138
+ {
139
+ title: t('menus.model_metadata.manage.fields.permissionName'), // 权限组名称
140
+ key: 'permissionName',
141
+ width: 160,
142
+ render: (_: unknown, row: UserPermissionRow) => {
143
+ const tpl = templateMap.get(row.permissionId)
144
+ return (
145
+ <span className="text-sm text-gray-900 dark:text-gray-100">
146
+ {tpl?.permissionName || '—'}
147
+ </span>
148
+ )
149
+ },
150
+ },
151
+ {
152
+ title: t('menus.model_metadata.manage.fields.permissionColumns'), // 模板允许的列名
153
+ key: 'permissionColumns',
154
+ minWidth: 200,
155
+ render: (_: unknown, row: UserPermissionRow) => {
156
+ const tpl = templateMap.get(row.permissionId)
157
+ return (
158
+ <span className="text-sm text-gray-600 dark:text-gray-300">
159
+ {formatPermissionColumns(tpl?.permissionColumns ?? [])}
160
+ </span>
161
+ )
162
+ },
163
+ },
164
+ {
165
+ title: t('menus.model_metadata.role.assign_users_permission_value'), // 权限值
166
+ key: 'permissionValue',
167
+ width: 240,
168
+ render: (_: unknown, row: UserPermissionRow) => {
169
+ if (!activeUser) return null
170
+ return (
171
+ <CxInput
172
+ className="w-[240px] max-w-full"
173
+ value={row.permissionValue}
174
+ onChange={(e) =>
175
+ onPermissionValueChange(
176
+ activeUser.userId,
177
+ row.metadataId,
178
+ row.permissionId,
179
+ e.target.value,
180
+ )
181
+ }
182
+ placeholder={t('menus.model_metadata.role.assign_users_permission_value')}
183
+ />
184
+ )
185
+ },
186
+ },
187
+ ]
188
+
189
+ return (
190
+ <section
191
+ key={mid}
192
+ ref={(el) => {
193
+ sectionRefs.current[mid] = el
194
+ }}
195
+ data-metadata-id={mid}
196
+ className={clsx(
197
+ 'scroll-mt-2 rounded-xl border bg-white p-4 shadow-sm dark:bg-gray-800',
198
+ activeMetadataId === mid
199
+ ? 'border-blue-300 ring-1 ring-blue-200/80 dark:border-blue-600 dark:ring-blue-900/50'
200
+ : 'border-gray-200 dark:border-gray-600',
201
+ )}
202
+ >
203
+ <h3 className="mb-3 text-sm font-semibold text-gray-900 dark:text-gray-100">
204
+ {t('menus.model_metadata.role.assign_users_library_title', {
205
+ subject: lib.subject || mid,
206
+ databaseName: lib.databaseName || '—',
207
+ })}
208
+ </h3>
209
+ {rows.length === 0 ? (
210
+ <p className="py-4 text-center text-sm text-gray-500">{t('common.no_data')}</p>
211
+ ) : (
212
+ <CxTable
213
+ className="min-w-0 w-full"
214
+ columns={columns}
215
+ dataSource={rows.map((r) => ({
216
+ ...r,
217
+ key: `${r.metadataId}::${r.permissionId}`,
218
+ }))}
219
+ pagination={false}
220
+ scroll={{ x: true }}
221
+ />
222
+ )}
223
+ </section>
224
+ )
225
+ }
226
+
227
+ const renderMetaNav = () => {
228
+ if (loading || !libraries.length || !activeUser) return null
229
+ return (
230
+ <div
231
+ className={clsx(
232
+ 'flex w-52 shrink-0 flex-col overflow-hidden border-r border-gray-200 bg-white dark:border-gray-700 dark:bg-gray-800',
233
+ hasLayoutLock && 'h-full min-h-0',
234
+ )}
235
+ >
236
+ <div className={sidePanelHeaderClass}>
237
+ <h2 className={sidePanelHeaderTitleClass}>
238
+ {t('menus.model_metadata.role.assign_users_nav_title')}
239
+ </h2>
240
+ </div>
241
+ <nav className={clsx(scrollColClass, 'p-2')}>
242
+ <ul className="space-y-1">
243
+ {libraries.map((lib) => {
244
+ const isNavActive = lib.metadataId === activeMetadataId
245
+ return (
246
+ <li key={lib.metadataId}>
247
+ <button
248
+ type="button"
249
+ onClick={() => scrollToMeta(lib.metadataId)}
250
+ className={clsx(
251
+ 'w-full rounded-l-lg border-r-[3px] px-2 py-2 text-left text-sm transition-colors',
252
+ isNavActive
253
+ ? 'border-r-blue-500 bg-blue-50 font-medium text-blue-900 dark:border-r-blue-400 dark:bg-blue-950/40 dark:text-blue-100'
254
+ : 'border-r-transparent text-gray-800 hover:bg-gray-100 dark:text-gray-100 dark:hover:bg-gray-700/50',
255
+ )}
256
+ >
257
+ <div className="line-clamp-2">{lib.subject || lib.metadataId}</div>
258
+ <div
259
+ className={clsx(
260
+ 'mt-0.5 line-clamp-1 text-xs',
261
+ isNavActive
262
+ ? 'text-blue-700/80 dark:text-blue-300/80'
263
+ : 'text-gray-500 dark:text-gray-400',
264
+ )}
265
+ >
266
+ {lib.databaseName ?? '—'}
267
+ </div>
268
+ </button>
269
+ </li>
270
+ )
271
+ })}
272
+ </ul>
273
+ </nav>
274
+ </div>
275
+ )
276
+ }
277
+
278
+ return (
279
+ <div className={rootLayoutClass}>
280
+ <div
281
+ className={clsx(
282
+ 'flex min-w-0 w-72 shrink-0 flex-col overflow-hidden rounded-xl border border-gray-200 bg-white shadow-sm dark:border-gray-700 dark:bg-gray-800',
283
+ hasLayoutLock && 'h-full min-h-0',
284
+ )}
285
+ >
286
+ <div className="flex min-h-[44px] shrink-0 items-center justify-between gap-2 border-b border-gray-200 bg-gray-50 px-3 py-2.5 dark:border-gray-700 dark:bg-gray-700/50">
287
+ <h2 className="min-w-0 flex-1 text-xs font-semibold uppercase tracking-wider text-gray-700 dark:text-gray-300">
288
+ {t('menus.model_metadata.role.assign_users_user_list')}
289
+ </h2>
290
+ <CxButton
291
+ type="primary"
292
+ size="small"
293
+ icon={<Plus size={14} />}
294
+ onClick={onAddUsersClick}
295
+ >
296
+ {t('menus.model_metadata.role.assign_users_add_user')}
297
+ </CxButton>
298
+ </div>
299
+ <div className={clsx(scrollColClass, 'p-2')}>
300
+ {loading ? (
301
+ <ul className="space-y-2 px-1 py-1">
302
+ {Array.from({ length: 5 }).map((_, i) => (
303
+ <li key={i}>
304
+ <CxSkeleton active paragraph={{ rows: 2, width: ['80%', '50%'] }} title={false} />
305
+ </li>
306
+ ))}
307
+ </ul>
308
+ ) : users.length === 0 ? (
309
+ <p className="px-2 py-6 text-center text-sm text-gray-500">
310
+ {t('menus.model_metadata.role.assign_users_no_user')}
311
+ </p>
312
+ ) : (
313
+ <ul className="space-y-1">
314
+ {users.map((u) => {
315
+ const isActive = u.userId === activeUserId
316
+ return (
317
+ <li key={u.userId}>
318
+ <div
319
+ className={clsx(
320
+ 'flex w-full items-stretch overflow-hidden rounded-lg border text-left text-sm transition-colors',
321
+ isActive
322
+ ? 'border-blue-300 bg-blue-50 text-blue-900 dark:border-blue-700 dark:bg-blue-950/40 dark:text-blue-100'
323
+ : 'border-transparent bg-transparent text-gray-800 hover:bg-gray-50 dark:text-gray-100 dark:hover:bg-gray-700/40',
324
+ )}
325
+ >
326
+ <button
327
+ type="button"
328
+ onClick={() => onActiveUserIdChange(u.userId)}
329
+ className="min-w-0 flex-1 px-3 py-2 text-left"
330
+ >
331
+ <div className="line-clamp-1 font-medium">
332
+ {u.userName || u.userId}
333
+ </div>
334
+ {u.userPhone ? (
335
+ <div className="mt-0.5 line-clamp-1 text-xs text-gray-500 dark:text-gray-400">
336
+ {u.userPhone}
337
+ </div>
338
+ ) : null}
339
+ </button>
340
+ <div className="flex shrink-0 items-center px-2">
341
+ <CxButton.Icon
342
+ type="default"
343
+ title={t('menus.model_metadata.role.assign_users_remove_user')}
344
+ icon={<Trash2 size={14} />}
345
+ onClick={() => onRemoveUser(u.userId)}
346
+ />
347
+ </div>
348
+ </div>
349
+ </li>
350
+ )
351
+ })}
352
+ </ul>
353
+ )}
354
+ </div>
355
+ </div>
356
+
357
+ <div
358
+ className={clsx(
359
+ 'flex min-w-0 flex-1 overflow-hidden rounded-xl border border-gray-200 bg-gray-50 dark:border-gray-700 dark:bg-gray-900/50',
360
+ hasLayoutLock && 'h-full min-h-0',
361
+ )}
362
+ >
363
+ {renderMetaNav()}
364
+
365
+ <div className="flex min-w-0 flex-1 flex-col overflow-hidden">
366
+ <div className={sidePanelHeaderClass}>
367
+ <h2 className={sidePanelHeaderTitleClass}>
368
+ {t('menus.model_metadata.role.assign_users_permission_panel')}
369
+ {activeUser?.userName ? ` — ${activeUser.userName}` : ''}
370
+ </h2>
371
+ </div>
372
+ <div ref={contentScrollRef} className={clsx(scrollColClass, 'p-3')}>
373
+ {loading ? (
374
+ <div className="space-y-4">
375
+ {Array.from({ length: 3 }).map((_, i) => (
376
+ <CxSkeleton key={i} active paragraph={{ rows: 4 }} />
377
+ ))}
378
+ </div>
379
+ ) : !activeUser ? (
380
+ <p className="py-10 text-center text-sm text-gray-500">
381
+ {t('menus.model_metadata.role.assign_users_no_user')}
382
+ </p>
383
+ ) : libraries.length === 0 ? (
384
+ <p className="py-10 text-center text-sm text-gray-500">{t('common.no_data')}</p>
385
+ ) : (
386
+ <div className="space-y-4">{libraries.map(renderLibrarySection)}</div>
387
+ )}
388
+ </div>
389
+ </div>
390
+ </div>
391
+ </div>
392
+ )
393
+ }