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,541 @@
1
+ import ConfirmDialog from '@/components/common/modal/confirm-dialog'
2
+ import UserSelectMultiModal from '@/components/common/modal/user-select-multi'
3
+ import { useAuth } from '@/contexts/AuthContext'
4
+ import {
5
+ bindUserDepartmentsAPI,
6
+ deleteDepartmentAPI,
7
+ getDepartmentTreeAPI,
8
+ getDepartmentUsersAPI,
9
+ unbindUserDepartmentsAPI
10
+ } from '@api/system/department'
11
+ import type { SearchButtonSize, TreeContext } from '@components/table-view'
12
+ import ListPageView, {
13
+ createListPageConfig,
14
+ toCxSearchButtonSize,
15
+ } from '@components/table-view'
16
+ import { CxButton, CxEmptyState } from '@cx-ui'
17
+ import { Edit, UserPlus } from 'lucide-react'
18
+ import React, { useEffect, useMemo, useState } from 'react'
19
+ import { useTranslation } from 'react-i18next'
20
+ import DetailModal from './modal/detail-modal'
21
+ import MemberRoleModal from './modal/member-role-modal'
22
+ import SubmitModal from './modal/submit-modal'
23
+
24
+ const useListPageConfig = createListPageConfig({
25
+ /** 不自动选根:挂载即拉成员表,避免未选部门时仍走「等选根」逻辑;点选部门后再带条件刷新 */
26
+ tree: { defaultSelectRoot: false ,searchField: 'departmentId'},
27
+ deleteAPI: deleteDepartmentAPI,
28
+ tableAPI: getDepartmentUsersAPI,
29
+ treeAPI: getDepartmentTreeAPI,
30
+ pageInfo: {
31
+ title: 'system.department.title',
32
+ subTitle: 'system.department.subtitle',
33
+ listTitle: 'system.department.list.members_title',
34
+ searchButtonSize: 'small',
35
+ },
36
+ searchFields: [
37
+ {
38
+ prop: 'keyword',
39
+ label: 'system.department.search.member_placeholder',
40
+ placeholder: 'system.department.search.member_placeholder',
41
+ },
42
+ ],
43
+ table: {
44
+ showPagination: true,
45
+ },
46
+ submitAction: {
47
+ permission: 'department:add',
48
+ submitType: 'modal',
49
+ labelKey: 'system.department.actions.create',
50
+ },
51
+ detailAction: {
52
+ detailType: 'modal',
53
+ labelKey: 'system.department.actions.detail',
54
+ permission: 'department:detail',
55
+ },
56
+ })
57
+
58
+ /**
59
+ * 部门管理页面的表格工具栏渲染函数
60
+ */
61
+ function renderTableActions(
62
+ {
63
+ tree,
64
+ }: {
65
+ searchButtonSize: SearchButtonSize
66
+ tree?: TreeContext
67
+ },
68
+ pageConfig: ReturnType<typeof useListPageConfig>,
69
+ hasPermission: (permission: string) => boolean,
70
+ submitModalRef: React.RefObject<{ open: (p?: any) => void } | null>,
71
+ ) {
72
+ const { t } = useTranslation()
73
+ const selectedDept = tree?.selectedNode as any | null
74
+ if (!selectedDept) return null
75
+ /** 有树列表中间工具栏:固定 34px,与默认 small(32) 区分 */
76
+ const treeToolbarIconPx = 14
77
+ const outlineBtnClass =
78
+ 'border-gray-300 dark:border-gray-600 text-gray-700 dark:text-gray-200 shadow-sm'
79
+
80
+ // 通过部门编码在 treeData 中定位,并回溯组装路径;tree 的第一级不计入显示/level
81
+ const getPathInfoByCode = (
82
+ departments: any[],
83
+ code: string | undefined,
84
+ ): { names: string; level: number } => {
85
+ const targetCode = String(code ?? '').trim()
86
+ if (!targetCode) return { names: '', level: 0 }
87
+
88
+ const walk = (nodes: any[], acc: any[] = []): any[] | null => {
89
+ for (const n of nodes || []) {
90
+ const nextAcc = [...acc, n]
91
+ if (String(n?.code ?? '').trim() === targetCode) return nextAcc
92
+ const children = n?.children
93
+ if (Array.isArray(children) && children.length) {
94
+ const found = walk(children, nextAcc)
95
+ if (found) return found
96
+ }
97
+ }
98
+ return null
99
+ }
100
+
101
+ const pathNodes = walk(departments, []) || []
102
+ // 不记录第一级
103
+ const effectiveNodes = pathNodes.slice(1)
104
+ const names = effectiveNodes.map((n) => n?.name ?? n?.code ?? '').filter(Boolean)
105
+ return {
106
+ names: names.join(' > '),
107
+ level: effectiveNodes.length,
108
+ }
109
+ }
110
+
111
+ const pathInfo = getPathInfoByCode(tree?.treeData ?? [], selectedDept.code)
112
+
113
+ return (
114
+ <>
115
+ <div>
116
+ <h1 className="text-2xl font-bold text-gray-900 dark:text-white flex items-center gap-2">
117
+ {selectedDept.name}
118
+ {/* {!selectedDept.is_active && (
119
+ <span className="text-xs bg-red-100 dark:bg-red-900/30 text-red-600 dark:text-red-400 px-2 py-0.5 rounded-full">
120
+ {t('system.department.inactive')}
121
+ </span>
122
+ )} */}
123
+ </h1>
124
+ <div className="text-sm text-gray-500 dark:text-gray-400 mt-1 flex gap-4">
125
+ {selectedDept.code && (
126
+ <span>
127
+ {t('system.department.fields.code')}: {selectedDept.code}
128
+ </span>
129
+ )}
130
+ <span>
131
+ {t('system.department.fields.level')}: {pathInfo.level}
132
+ </span>
133
+ <span>
134
+ {t('system.department.fields.path')}:{' '}
135
+ {pathInfo.names}
136
+ </span>
137
+ </div>
138
+ {selectedDept.description && (
139
+ <p className="text-gray-600 dark:text-gray-400 mt-2 text-sm">
140
+ {selectedDept.description}
141
+ </p>
142
+ )}
143
+ </div>
144
+ <div className="flex gap-2 flex-wrap items-center">
145
+ {/* 新增子部门按钮 */}
146
+ {hasPermission('department:add') && (
147
+ <CxButton.Add
148
+ type="default"
149
+ height={34}
150
+ iconSize={treeToolbarIconPx}
151
+ onClick={() => {
152
+ submitModalRef.current?.open({
153
+ opeType: 'add',
154
+ data: selectedDept,
155
+ })
156
+ }}
157
+ className={outlineBtnClass}>
158
+ {t('system.department.actions.add_child')}
159
+ </CxButton.Add>
160
+ )}
161
+ {/* 编辑部门按钮 */}
162
+ {hasPermission('department:edit') && (
163
+ <CxButton
164
+ type="default"
165
+ height={34}
166
+ icon={<Edit size={treeToolbarIconPx} />}
167
+ className={outlineBtnClass}
168
+ onClick={() => {
169
+ submitModalRef.current?.open({
170
+ opeType: 'edit',
171
+ opeId: selectedDept.id,
172
+ data: selectedDept,
173
+ })
174
+ }}>
175
+ {t('system.department.actions.edit')}
176
+ </CxButton>
177
+ )}
178
+ {/* 删除部门按钮 */}
179
+ {hasPermission('department:delete') && (
180
+ <CxButton.Delete
181
+ height={34}
182
+ iconSize={treeToolbarIconPx}
183
+ onClick={() => pageConfig.onDelete?.(selectedDept)}>
184
+ {t('system.department.actions.delete')}
185
+ </CxButton.Delete>
186
+ )}
187
+ </div>
188
+ </>
189
+ )
190
+ }
191
+
192
+ function IndexView() {
193
+ const { t } = useTranslation()
194
+ const { hasPermission, user, refreshUser } = useAuth()
195
+ const canViewTree = hasPermission('department:tree')
196
+ const canViewMemberPage = hasPermission('department:page')
197
+ const canAddDepartment = hasPermission('department:add')
198
+ const canAddMember = hasPermission('department:member-add')
199
+ const submitModalRef = React.useRef<{ open: (p?: any) => void } | null>(null)
200
+ const memberRoleModalRef =
201
+ React.useRef<{ open: (p?: any) => void; close: () => void } | null>(null)
202
+ const userSelectModalRef =
203
+ React.useRef<{ open: (p: { departmentId: number | string }) => void } | null>(
204
+ null,
205
+ )
206
+ const [treeReloadSignal, setTreeReloadSignal] = useState(0)
207
+ const basePageConfig = useListPageConfig()
208
+ const pageConfig = {
209
+ ...basePageConfig,
210
+ treeReloadSignal,
211
+ // treeAPI: canViewTree ? getDepartmentTreeAPI : undefined,
212
+ // tableAPI: canViewMemberPage ? getDepartmentUsersAPI : undefined,
213
+ }
214
+ const [removeConfirmOpen, setRemoveConfirmOpen] = useState(false)
215
+ const [pendingRemoveRow, setPendingRemoveRow] = useState<any>(null)
216
+ const [roleConfirmOpen, setRoleConfirmOpen] = useState(false)
217
+ const [pendingRoleRow, setPendingRoleRow] = useState<any>(null)
218
+
219
+ /**
220
+ * 页面加载后打印当前用户的部门信息列表。
221
+ */
222
+ // useEffect(() => {
223
+ // console.log(
224
+ // '[department] 当前用户 departmentUserInfoVOList:',
225
+ // (user as any)?.departmentUserInfoVOList,
226
+ // )
227
+ // }, [])
228
+
229
+ /**
230
+ * 获取当前选中部门 ID。
231
+ */
232
+ const getSelectedDepartmentId = () => {
233
+ const searchDepartmentId = (pageConfig as any)?.searchFormData?.departmentId
234
+ if (searchDepartmentId !== undefined && searchDepartmentId !== null && searchDepartmentId !== '') {
235
+ return searchDepartmentId
236
+ }
237
+ const treeCtx = (pageConfig as any)?.treeContext as TreeContext | undefined
238
+ return treeCtx?.selectedNode?.id
239
+ }
240
+
241
+ /**
242
+ * 根据部门 ID 获取当前用户在该部门的角色。
243
+ */
244
+ const getCurrentUserDepartmentRole = (departmentId: any) => {
245
+ const departmentUserInfoVOList = (user as any)?.departmentUserInfoVOList || []
246
+ const matched = departmentUserInfoVOList.find(
247
+ (item: any) => item?.departmentId == departmentId,
248
+ )
249
+ return matched?.departmentRole
250
+ }
251
+
252
+ /**
253
+ * 判断当前用户是否可设置部门角色(仅管理员/负责人)。
254
+ */
255
+ const canSetDepartmentRole = (departmentId: any) => {
256
+ const role = getCurrentUserDepartmentRole(departmentId)
257
+ return role === 20 || role === 30 || role === '20' || role === '30'
258
+ }
259
+
260
+ if (!canViewTree) {
261
+ return <CxEmptyState message={t('common.page_error.401.message')} />
262
+ }
263
+
264
+ const columns = useMemo(
265
+ () => [
266
+ {
267
+ title: 'system.department.fields.member',
268
+ key: 'userName',
269
+ render: (_: any, data: any) => (
270
+ <div className="flex items-center gap-3">
271
+ <div className="w-8 h-8 rounded-full bg-blue-100 dark:bg-blue-900/30 flex items-center justify-center text-blue-600 dark:text-blue-400 font-bold uppercase">
272
+ {data.userName?.[0] || data.userEmail?.[0]}
273
+ </div>
274
+ <div>
275
+ <div className="text-sm font-medium">
276
+ {data.userName || data.userEmail}
277
+ </div>
278
+ <div className="text-xs text-gray-500 dark:text-gray-400">
279
+ {data.userEmail}
280
+ </div>
281
+ </div>
282
+ </div>
283
+ ),
284
+ },
285
+ {
286
+ title: 'system.department.fields.role',
287
+ key: 'departmentRole',
288
+ render: (value: any, data: any) => {
289
+ const v = Number(value)
290
+ if (v === 10) return '成员'
291
+ if (v === 20) return '管理员'
292
+ if (v === 30) return '负责人'
293
+ return value ?? '-'
294
+ },
295
+ },
296
+ {
297
+ title: 'system.department.fields.joined',
298
+ key: 'joinTime',
299
+ render: (value: string, data: any) =>
300
+ value ? new Date(value).toLocaleDateString() : '',
301
+ },
302
+ {
303
+ title: '操作',
304
+ key: 'action',
305
+ width: 180,
306
+ render: (_: any, data: any) => {
307
+ const departmentId = getSelectedDepartmentId()
308
+ const canRemoveMember = hasPermission('department:member-remove')
309
+ const canSetMemberRole = canSetDepartmentRole(departmentId)
310
+ if (!canRemoveMember && !canSetMemberRole) return null
311
+ return (
312
+ <div className="flex gap-2">
313
+ {canSetMemberRole && (
314
+ <CxButton
315
+ type="link"
316
+ onClick={() => {
317
+ openRoleConfirm(data)
318
+ }}>
319
+ 设置部门角色
320
+ </CxButton>
321
+ )}
322
+ {canRemoveMember && (
323
+ <CxButton
324
+ type="link"
325
+ className="!text-[#ef4444] hover:!text-[#b91c1c]"
326
+ onClick={() => openRemoveConfirm(data)}>
327
+ 移除
328
+ </CxButton>
329
+ )}
330
+ </div>
331
+ )
332
+ },
333
+ },
334
+ ],
335
+ [t, pageConfig, hasPermission, user],
336
+ )
337
+
338
+ // 添加成员
339
+ const handleAddMembersOk = async (data: any) => {
340
+ const deptId = data?.departmentId
341
+ if (deptId == null || deptId === '') return false
342
+ const users = Array.isArray(data?.selectedUsers) ? data.selectedUsers : []
343
+ if (users.length === 0) return false
344
+
345
+ const userItems = users
346
+ .map((u: any) => ({ userId: u?.id, departmentRole: data?.role }))
347
+ .filter((x: any) => x.userId !== undefined && x.userId !== null && x.userId !== '')
348
+
349
+ if (userItems.length === 0) return false
350
+
351
+ await bindUserDepartmentsAPI({
352
+ departmentId: deptId,
353
+ userItems,
354
+ })
355
+ }
356
+ // 移除成员
357
+ const handleRemoveMember = async (row: any) => {
358
+ const treeCtx = (pageConfig as any)?.treeContext as TreeContext | undefined
359
+ const dept = treeCtx?.selectedNode as any | null
360
+ const deptId = dept?.id
361
+ if (deptId == null || String(deptId).trim() === '') return false
362
+
363
+ const userId = row?.userId
364
+ if (userId == null || String(userId).trim() === '') return false
365
+
366
+ await unbindUserDepartmentsAPI({
367
+ departmentId: deptId,
368
+ userIds: [ userId ],
369
+ })
370
+
371
+ await pageConfig.fetchTableData?.()
372
+ }
373
+
374
+ /**
375
+ * 设置成员角色后刷新列表;若改的是当前登录用户则同步刷新用户信息以更新按钮权限。
376
+ */
377
+ const handleAfterSetMemberRole = async (payload?: {
378
+ row?: any
379
+ role?: number
380
+ departmentId?: number | string
381
+ }) => {
382
+ await pageConfig.fetchTableData?.()
383
+ const targetUserId = payload?.row?.userId
384
+ const currentUserId = (user as any)?.id
385
+ if (
386
+ targetUserId != null &&
387
+ targetUserId !== '' &&
388
+ currentUserId != null &&
389
+ currentUserId !== '' &&
390
+ targetUserId == currentUserId
391
+ ) {
392
+ await refreshUser()
393
+ }
394
+ }
395
+
396
+ const openRemoveConfirm = (row: any) => {
397
+ setPendingRemoveRow(row)
398
+ setRemoveConfirmOpen(true)
399
+ }
400
+
401
+ /**
402
+ * 打开设置部门角色前的确认弹框。
403
+ */
404
+ const openRoleConfirm = (row: any) => {
405
+ setPendingRoleRow(row)
406
+ setRoleConfirmOpen(true)
407
+ }
408
+
409
+ const closeRemoveConfirm = () => {
410
+ setRemoveConfirmOpen(false)
411
+ setPendingRemoveRow(null)
412
+ }
413
+
414
+ /**
415
+ * 关闭设置部门角色前的确认弹框。
416
+ */
417
+ const closeRoleConfirm = () => {
418
+ setRoleConfirmOpen(false)
419
+ setPendingRoleRow(null)
420
+ }
421
+
422
+ const confirmRemove = async () => {
423
+ if (!pendingRemoveRow) return
424
+ await handleRemoveMember(pendingRemoveRow)
425
+ closeRemoveConfirm()
426
+ }
427
+
428
+ /**
429
+ * 确认后再打开设置部门角色弹框。
430
+ */
431
+ const confirmRoleSetting = () => {
432
+ if (!pendingRoleRow) return
433
+ const departmentId = getSelectedDepartmentId()
434
+ memberRoleModalRef.current?.open({ row: pendingRoleRow, departmentId })
435
+ closeRoleConfirm()
436
+ }
437
+
438
+ pageConfig.onAdd = () => {
439
+ submitModalRef.current?.open({ opeType: 'add' })
440
+ }
441
+ pageConfig.afterDelete = async (record: any) => {
442
+ if (typeof basePageConfig.afterDelete === 'function') {
443
+ await basePageConfig.afterDelete(record)
444
+ }
445
+ setTreeReloadSignal((n) => n + 1)
446
+ }
447
+
448
+ return (
449
+ <>
450
+ <ListPageView
451
+ columns={columns}
452
+ headerActions={
453
+ <div>
454
+ {/* 新增部门 */}
455
+ {canAddDepartment && (
456
+ <CxButton.Add
457
+ onClick={() => {
458
+ submitModalRef.current?.open({ opeType: 'add' })
459
+ }}
460
+ >
461
+ {t('system.department.actions.create')}
462
+ </CxButton.Add>
463
+ )}
464
+ </div>
465
+ }
466
+ pageConfig={pageConfig}
467
+ searchAreaExtra={({ tree, searchButtonSize }) => {
468
+ const dept = tree?.selectedNode as {
469
+ id?: number | string
470
+ _id?: number | string
471
+ } | null
472
+ const deptId = dept?.id
473
+ if (
474
+ deptId === undefined ||
475
+ deptId === null ||
476
+ String(deptId).trim() === ''
477
+ ) {
478
+ return null
479
+ }
480
+ if (!canAddMember) return null
481
+ return (
482
+ <CxButton.Add
483
+ size={toCxSearchButtonSize(searchButtonSize)}
484
+ icon={<UserPlus size={14} />}
485
+ onClick={() =>
486
+ userSelectModalRef.current?.open({ departmentId: deptId })
487
+ }>
488
+ {t('system.department.actions.add_member')}
489
+ </CxButton.Add>
490
+ )
491
+ }}
492
+ tableActions={(ctx) =>
493
+ renderTableActions(ctx, pageConfig, hasPermission, submitModalRef)
494
+ }
495
+ detailModalComponent={DetailModal as any}
496
+ />
497
+ <SubmitModal
498
+ ref={submitModalRef}
499
+ onOk={async () => {
500
+ await pageConfig.fetchTableData?.()
501
+ setTreeReloadSignal((n) => n + 1)
502
+ }}
503
+ onCancel={() => {
504
+ // 取消时的清理逻辑(可选)
505
+ }}
506
+ />
507
+ <UserSelectMultiModal
508
+ onOk={handleAddMembersOk}
509
+ ref={userSelectModalRef}
510
+ onAfterAdd={async () => {
511
+ await pageConfig.fetchTableData?.()
512
+ }}
513
+ />
514
+ <MemberRoleModal
515
+ ref={memberRoleModalRef}
516
+ onOk={handleAfterSetMemberRole}
517
+ />
518
+ <ConfirmDialog
519
+ isOpen={roleConfirmOpen}
520
+ title={t('common.confirm')}
521
+ message="确认设置该成员部门角色吗?"
522
+ confirmText={t('common.confirm')}
523
+ cancelText={t('common.cancel')}
524
+ onClose={closeRoleConfirm}
525
+ onConfirm={confirmRoleSetting}
526
+ />
527
+ <ConfirmDialog
528
+ isOpen={removeConfirmOpen}
529
+ title={t('common.confirm')}
530
+ message="确认移除该成员吗?"
531
+ confirmText={t('common.confirm')}
532
+ cancelText={t('common.cancel')}
533
+ onClose={closeRemoveConfirm}
534
+ onConfirm={confirmRemove}
535
+ danger
536
+ />
537
+ </>
538
+ )
539
+ }
540
+
541
+ export default IndexView
@@ -0,0 +1,94 @@
1
+ import React, { forwardRef, useImperativeHandle, useState } from 'react'
2
+ import { Descriptions } from 'antd'
3
+ import { useTranslation } from 'react-i18next'
4
+
5
+ import { CxModal, CxTag, useCxModal } from '@cx-ui'
6
+ import { getDepartmentDetailAPI } from '@/api/system/department'
7
+ import { Spin } from 'antd'
8
+
9
+ const DetailModal = forwardRef(function DetailModal(_props, ref) {
10
+ const { t } = useTranslation()
11
+ const [loading, setLoading] = useState(false)
12
+
13
+ const modal = useCxModal({
14
+ onOpen: async (payload) => {
15
+ const opeId = payload.opeId ?? payload.data?.id
16
+ setLoading(true)
17
+ modal.setFormData(undefined)
18
+ if (opeId == null || `${opeId}`.trim() === '') {
19
+ setLoading(false)
20
+ return
21
+ }
22
+ try {
23
+ const res = await getDepartmentDetailAPI({ id: opeId })
24
+ modal.setFormData(res?.data)
25
+ } finally {
26
+ setLoading(false)
27
+ }
28
+ },
29
+ onClose: () => {
30
+ setLoading(false)
31
+ },
32
+ })
33
+
34
+ const { isOpen, close } = modal
35
+ const record = modal.formData
36
+
37
+ useImperativeHandle(ref, () => ({
38
+ open: modal.open,
39
+ close,
40
+ }), [modal.open, close])
41
+
42
+ return (
43
+ <CxModal
44
+ title={`${t('system.department.title')} - ${t('system.department.actions.detail')}`}
45
+ isOpen={isOpen}
46
+ onClose={close}
47
+ opeType="detail">
48
+ {loading ? (
49
+ <div className="flex justify-center py-10">
50
+ <Spin />
51
+ </div>
52
+ ) : (
53
+ <Descriptions column={1} size="small" bordered>
54
+ <Descriptions.Item label={t('system.department.fields.name')}>
55
+ {record?.name || '-'}
56
+ </Descriptions.Item>
57
+ <Descriptions.Item label={t('system.department.fields.code')}>
58
+ {record?.code || '-'}
59
+ </Descriptions.Item>
60
+ <Descriptions.Item label={t('common.status')}>
61
+ {(() => {
62
+ const active = record?.is_active
63
+ if (active === undefined || active === null) return '-'
64
+ const v = active === true || Number(active) === 1 ? 1 : 0
65
+ return <CxTag.Enabled value={v} />
66
+ })()}
67
+ </Descriptions.Item>
68
+ <Descriptions.Item label={t('system.department.fields.level')}>
69
+ {record?.level ?? '-'}
70
+ </Descriptions.Item>
71
+ <Descriptions.Item label={t('system.department.fields.path')}>
72
+ {record?.path || '-'}
73
+ </Descriptions.Item>
74
+ <Descriptions.Item label={t('common.description')}>
75
+ {record?.description || '-'}
76
+ </Descriptions.Item>
77
+ {record?.created_at && (
78
+ <Descriptions.Item label={t('common.created_at')}>
79
+ {new Date(record.created_at).toLocaleString()}
80
+ </Descriptions.Item>
81
+ )}
82
+ {record?.updated_at && (
83
+ <Descriptions.Item label={t('common.updated_at')}>
84
+ {new Date(record.updated_at).toLocaleString()}
85
+ </Descriptions.Item>
86
+ )}
87
+ </Descriptions>
88
+ )}
89
+ </CxModal>
90
+ )
91
+ })
92
+
93
+ export default DetailModal
94
+