cx-chat 0.0.2 → 0.0.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (406) hide show
  1. package/dist/cx-chat.css +1 -0
  2. package/dist/cx-chat.es.js +68257 -0
  3. package/dist/cx-chat.umd.js +492 -0
  4. package/package.json +27 -3
  5. package/.cursor/rules/i18n-cn-gloss-comments.mdc +0 -31
  6. package/.cursor/rules/list-page-view-pageconfig.mdc +0 -32
  7. package/.cursor/rules/no-over-defensive-programming.mdc +0 -90
  8. package/.cursor/rules/requirement-description-for-agent.mdc +0 -33
  9. package/.cursor/rules/use-showToast-not-antd-message.mdc +0 -28
  10. package/.docker/Dockerfile +0 -7
  11. package/.env +0 -9
  12. package/.env.development +0 -7
  13. package/.env.production +0 -7
  14. package/.gitlab-ci/docker-build.yaml +0 -28
  15. package/.gitlab-ci/k8s-deploy-dev-master.yaml +0 -42
  16. package/.gitlab-ci/npm-build.yaml +0 -17
  17. package/.gitlab-ci.yml +0 -8
  18. package/.k8s/0-namespace.yaml +0 -6
  19. package/.k8s/1-configmap-web.yaml +0 -7
  20. package/.k8s/1-nginx-conf-dev.yaml +0 -110
  21. package/.k8s/2-deployment.yaml +0 -27
  22. package/.k8s/3-service.yaml +0 -16
  23. package/.k8s/4-ingress-dev.yaml +0 -30
  24. package/.lingma/rules/use-showToast-not-antd-message.md +0 -34
  25. package/.nginx/nginx.conf +0 -52
  26. package/.prettierrc +0 -9
  27. package/eslint.config.js +0 -32
  28. package/index.html +0 -13
  29. package/postcss.config.js +0 -6
  30. package/src/App.tsx +0 -96
  31. package/src/_doc/0.docs-overview.md +0 -28
  32. package/src/_doc/cx-ui/0.docs-overview.md +0 -30
  33. package/src/_doc/cx-ui/comp.1.cx-ui-overview.md +0 -82
  34. package/src/_doc/cx-ui/comp.2.cx-modal.md +0 -82
  35. package/src/_doc/cx-ui/comp.3.cx-button.md +0 -89
  36. package/src/_doc/cx-ui/comp.4.cx-form.md +0 -72
  37. package/src/_doc/cx-ui/comp.5.cx-fields.md +0 -76
  38. package/src/_doc/cx-ui/comp.6.cx-tag.md +0 -57
  39. package/src/_doc/cx-ui/comp.7.cx-empty-state.md +0 -29
  40. package/src/_doc/meta/0.docs-overview.md +0 -24
  41. package/src/_doc/meta/comp.1.enum-runtime.md +0 -33
  42. package/src/_doc/meta/comp.2.dict-runtime.md +0 -39
  43. package/src/_doc/router/0.docs-overview.md +0 -14
  44. package/src/_doc/router/guide.1.menu-component-config.md +0 -181
  45. package/src/_doc/router/guide.2.router-auto-registration.md +0 -114
  46. package/src/_doc/table-view/0.docs-overview.md +0 -30
  47. package/src/_doc/table-view/comp.1.table-view.md +0 -542
  48. package/src/_doc/table-view/props.1.create-table-view-config.md +0 -193
  49. package/src/_doc/table-view/props.2.table-view-search-fields.md +0 -106
  50. package/src/api/_mock/README.md +0 -340
  51. package/src/api/_mock/api.ts +0 -1642
  52. package/src/api/_mock/bundle-shim.ts +0 -16
  53. package/src/api/_mock/handler-shim.ts +0 -6
  54. package/src/api/_mock/handler.ts +0 -458
  55. package/src/api/_mock/index.ts +0 -711
  56. package/src/api/_mock/interceptor.ts +0 -15
  57. package/src/api/_mock/mod.ts +0 -12
  58. package/src/api/_mock/utils.ts +0 -65
  59. package/src/api/base/memory.js +0 -24
  60. package/src/api/chat.js +0 -210
  61. package/src/api/common/auth.js +0 -70
  62. package/src/api/menus/business-rules.js +0 -76
  63. package/src/api/menus/feedback.js +0 -102
  64. package/src/api/menus/knowledge.js +0 -159
  65. package/src/api/menus/model-metadata/manage.js +0 -70
  66. package/src/api/menus/model-metadata/role.js +0 -50
  67. package/src/api/menus/model-metadata/training-detail-mock-data.js +0 -569
  68. package/src/api/menus/model-metadata/training.js +0 -28
  69. package/src/api/menus/skill.js +0 -40
  70. package/src/api/system/agent-config.js +0 -16
  71. package/src/api/system/department.js +0 -94
  72. package/src/api/system/dict.js +0 -86
  73. package/src/api/system/menu.js +0 -37
  74. package/src/api/system/permission.js +0 -26
  75. package/src/api/system/role.js +0 -34
  76. package/src/api/system/sys-config.js +0 -16
  77. package/src/api/system/sys-log.js +0 -17
  78. package/src/api/system/user.js +0 -75
  79. package/src/api/upload.js +0 -39
  80. package/src/assets/react.svg +0 -1
  81. package/src/components/auth/current-user-avatar.tsx +0 -77
  82. package/src/components/common/code-view.tsx +0 -149
  83. package/src/components/common/detail-link.tsx +0 -67
  84. package/src/components/common/error-boundary.tsx +0 -98
  85. package/src/components/common/language-switcher.tsx +0 -91
  86. package/src/components/common/lite-table/index.tsx +0 -135
  87. package/src/components/common/md-editor.tsx +0 -126
  88. package/src/components/common/modal/confirm-dialog.tsx +0 -113
  89. package/src/components/common/modal/dep-user-select-multi.tsx +0 -324
  90. package/src/components/common/modal/dep-user-select.tsx +0 -249
  91. package/src/components/common/modal/user-select-multi.tsx +0 -266
  92. package/src/components/common/pagination.tsx +0 -472
  93. package/src/components/common/path.tsx +0 -175
  94. package/src/components/common/system-logo-mark.tsx +0 -48
  95. package/src/components/cx-ui/button/index.less +0 -208
  96. package/src/components/cx-ui/button/index.tsx +0 -611
  97. package/src/components/cx-ui/checkbox/index.tsx +0 -78
  98. package/src/components/cx-ui/date-picker/index.less +0 -17
  99. package/src/components/cx-ui/date-picker/index.tsx +0 -193
  100. package/src/components/cx-ui/drawer/index.tsx +0 -47
  101. package/src/components/cx-ui/empty-state/index.tsx +0 -20
  102. package/src/components/cx-ui/floating-shell/CxFloatingShell.tsx +0 -89
  103. package/src/components/cx-ui/floating-shell/cx-floating-shell.less +0 -283
  104. package/src/components/cx-ui/floating-shell/has-floating-value.ts +0 -41
  105. package/src/components/cx-ui/form/CxForm.tsx +0 -15
  106. package/src/components/cx-ui/form/index.tsx +0 -20
  107. package/src/components/cx-ui/form-item/index.less +0 -26
  108. package/src/components/cx-ui/form-item/index.tsx +0 -36
  109. package/src/components/cx-ui/index.ts +0 -70
  110. package/src/components/cx-ui/input/auto-complete.tsx +0 -134
  111. package/src/components/cx-ui/input/index.tsx +0 -259
  112. package/src/components/cx-ui/input-number/index.jsx +0 -66
  113. package/src/components/cx-ui/modal/index.jsx +0 -212
  114. package/src/components/cx-ui/modal/index.less +0 -144
  115. package/src/components/cx-ui/modal/useCxModal.ts +0 -125
  116. package/src/components/cx-ui/multi-select/index.jsx +0 -74
  117. package/src/components/cx-ui/multi-select/index.less +0 -40
  118. package/src/components/cx-ui/multi-select/index2.tsx +0 -361
  119. package/src/components/cx-ui/radio/index.tsx +0 -33
  120. package/src/components/cx-ui/range-picker/index.less +0 -65
  121. package/src/components/cx-ui/range-picker/index.tsx +0 -219
  122. package/src/components/cx-ui/select/index.less +0 -34
  123. package/src/components/cx-ui/select/index.tsx +0 -196
  124. package/src/components/cx-ui/skeleton/index.tsx +0 -12
  125. package/src/components/cx-ui/steps/index.tsx +0 -14
  126. package/src/components/cx-ui/styles/_tokens.less +0 -79
  127. package/src/components/cx-ui/styles/index.less +0 -246
  128. package/src/components/cx-ui/switch/index.less +0 -106
  129. package/src/components/cx-ui/switch/index.tsx +0 -120
  130. package/src/components/cx-ui/table/index.less +0 -160
  131. package/src/components/cx-ui/table/index.tsx +0 -152
  132. package/src/components/cx-ui/tabs/index.less +0 -15
  133. package/src/components/cx-ui/tabs/index.tsx +0 -34
  134. package/src/components/cx-ui/tag/index.less +0 -51
  135. package/src/components/cx-ui/tag/index.tsx +0 -140
  136. package/src/components/cx-ui/timeline/index.tsx +0 -14
  137. package/src/components/cx-ui/tooltip/index.tsx +0 -67
  138. package/src/components/cx-ui/tree/index.tsx +0 -193
  139. package/src/components/cx-ui/tree-select/index.jsx +0 -91
  140. package/src/components/cx-ui/tree-select/index.less +0 -27
  141. package/src/components/cx-ui/upload-file/index.less +0 -223
  142. package/src/components/cx-ui/upload-file/index.tsx +0 -640
  143. package/src/components/cx-ui/upload-img/index.tsx +0 -291
  144. package/src/components/layout/components/Header.tsx +0 -216
  145. package/src/components/layout/components/Sidebar.tsx +0 -717
  146. package/src/components/layout/index.tsx +0 -95
  147. package/src/components/table-view/components/search-area.tsx +0 -411
  148. package/src/components/table-view/components/table-view-config.tsx +0 -528
  149. package/src/components/table-view/components/table-view.types.ts +0 -478
  150. package/src/components/table-view/components/tree-api-normalize.ts +0 -38
  151. package/src/components/table-view/components/tree-data-annotate.ts +0 -31
  152. package/src/components/table-view/components/tree-sidebar.tsx +0 -74
  153. package/src/components/table-view/index.tsx +0 -61
  154. package/src/components/table-view/list-page-view.tsx +0 -1049
  155. package/src/components/table-view/select-table-view.tsx +0 -1094
  156. package/src/components/table-view/styles/select-table-view.less +0 -51
  157. package/src/config/default-system-name.ts +0 -9
  158. package/src/config/system.ts +0 -165
  159. package/src/constants/countryCodes.ts +0 -3
  160. package/src/contexts/AuthContext.tsx +0 -256
  161. package/src/contexts/ChatContext.tsx +0 -839
  162. package/src/contexts/MenuContext.tsx +0 -62
  163. package/src/contexts/ToastContext.tsx +0 -181
  164. package/src/hooks/useCopyToClipboard.ts +0 -47
  165. package/src/hooks/useModalSubmit.ts +0 -104
  166. package/src/hooks/useRouter.ts +0 -240
  167. package/src/hooks/useStepForm.ts +0 -46
  168. package/src/hooks/useStickyHeader.ts +0 -42
  169. package/src/hooks/useThreadActions.ts +0 -105
  170. package/src/hooks/useUserPreferences.ts +0 -101
  171. package/src/http/axios.js +0 -372
  172. package/src/http/mock.interceptor.ts +0 -9
  173. package/src/http/obfuscationKey.ts +0 -41
  174. package/src/i18n.ts +0 -60
  175. package/src/index.js +0 -1
  176. package/src/index.less +0 -169
  177. package/src/locales/en/auth.ts +0 -70
  178. package/src/locales/en/base/memory.ts +0 -28
  179. package/src/locales/en/base/settings.ts +0 -41
  180. package/src/locales/en/chat.ts +0 -40
  181. package/src/locales/en/common.ts +0 -173
  182. package/src/locales/en/enum.ts +0 -27
  183. package/src/locales/en/menus/business-rules.ts +0 -48
  184. package/src/locales/en/menus/feedback.ts +0 -62
  185. package/src/locales/en/menus/knowledge.ts +0 -120
  186. package/src/locales/en/menus/model-metadata/index.ts +0 -10
  187. package/src/locales/en/menus/model-metadata/manage.ts +0 -151
  188. package/src/locales/en/menus/model-metadata/role.ts +0 -48
  189. package/src/locales/en/menus/model-metadata/training.ts +0 -65
  190. package/src/locales/en/menus/skill.ts +0 -34
  191. package/src/locales/en/system/agent-config.ts +0 -34
  192. package/src/locales/en/system/department.ts +0 -68
  193. package/src/locales/en/system/dict.ts +0 -44
  194. package/src/locales/en/system/menu.ts +0 -45
  195. package/src/locales/en/system/permission.ts +0 -89
  196. package/src/locales/en/system/role.ts +0 -25
  197. package/src/locales/en/system/sys-config.ts +0 -33
  198. package/src/locales/en/system/sys-log.ts +0 -38
  199. package/src/locales/en/system/user.ts +0 -113
  200. package/src/locales/en.ts +0 -68
  201. package/src/locales/zh/auth.ts +0 -70
  202. package/src/locales/zh/base/memory.ts +0 -29
  203. package/src/locales/zh/base/settings.ts +0 -41
  204. package/src/locales/zh/chat.ts +0 -47
  205. package/src/locales/zh/common.ts +0 -178
  206. package/src/locales/zh/enum.ts +0 -28
  207. package/src/locales/zh/menus/business-rules.ts +0 -47
  208. package/src/locales/zh/menus/feedback.ts +0 -62
  209. package/src/locales/zh/menus/knowledge.ts +0 -117
  210. package/src/locales/zh/menus/model-metadata/index.ts +0 -10
  211. package/src/locales/zh/menus/model-metadata/manage.ts +0 -151
  212. package/src/locales/zh/menus/model-metadata/role.ts +0 -47
  213. package/src/locales/zh/menus/model-metadata/training.ts +0 -64
  214. package/src/locales/zh/menus/skill.ts +0 -34
  215. package/src/locales/zh/system/agent-config.ts +0 -33
  216. package/src/locales/zh/system/department.ts +0 -69
  217. package/src/locales/zh/system/dict.ts +0 -44
  218. package/src/locales/zh/system/menu.ts +0 -47
  219. package/src/locales/zh/system/permission.ts +0 -94
  220. package/src/locales/zh/system/role.ts +0 -25
  221. package/src/locales/zh/system/sys-config.ts +0 -32
  222. package/src/locales/zh/system/sys-log.ts +0 -38
  223. package/src/locales/zh/system/user.ts +0 -114
  224. package/src/locales/zh.ts +0 -67
  225. package/src/main.tsx +0 -50
  226. package/src/meta/const/index.ts +0 -40
  227. package/src/meta/index-dict.ts +0 -56
  228. package/src/meta/index-enum.ts +0 -95
  229. package/src/meta/index.ts +0 -14
  230. package/src/meta/module/dict-data/runtime.ts +0 -199
  231. package/src/meta/module/dict-data/types.ts +0 -17
  232. package/src/meta/module/enum-data/runtime.ts +0 -75
  233. package/src/meta/module/enum-data/types.ts +0 -18
  234. package/src/router/index.tsx +0 -312
  235. package/src/styles/AntdThemeProvider.tsx +0 -40
  236. package/src/styles/antd-theme.ts +0 -20
  237. package/src/styles/global.less +0 -107
  238. package/src/styles/variable.less +0 -103
  239. package/src/types/feedback.ts +0 -43
  240. package/src/types/index.ts +0 -85
  241. package/src/types/menu.ts +0 -43
  242. package/src/utils/aesUtil.ts +0 -123
  243. package/src/utils/chatUtils.ts +0 -524
  244. package/src/utils/cn.ts +0 -6
  245. package/src/utils/crypto.ts +0 -164
  246. package/src/utils/date.ts +0 -72
  247. package/src/utils/file-icons.tsx +0 -79
  248. package/src/utils/index.ts +0 -168
  249. package/src/utils/markdown-math-plugins.ts +0 -21
  250. package/src/utils/menuI18n.ts +0 -305
  251. package/src/utils/menuRouteRegistry.ts +0 -78
  252. package/src/utils/permission-crud.ts +0 -147
  253. package/src/utils/routeConfig.ts +0 -350
  254. package/src/utils/storage.ts +0 -135
  255. package/src/utils/toastBridge.ts +0 -26
  256. package/src/utils/url.ts +0 -38
  257. package/src/utils/validation.ts +0 -16
  258. package/src/views/auth/auth-code/index.less +0 -169
  259. package/src/views/auth/auth-code/index.module.less +0 -174
  260. package/src/views/auth/auth-code/index.tsx +0 -233
  261. package/src/views/auth/login.tsx +0 -498
  262. package/src/views/auth/register.tsx +0 -388
  263. package/src/views/base/memory/index.tsx +0 -136
  264. package/src/views/base/memory/modal/detail-modal.tsx +0 -89
  265. package/src/views/base/memory/modal/submit-modal.tsx +0 -134
  266. package/src/views/base/settings/index.tsx +0 -657
  267. package/src/views/chat/chat.less +0 -323
  268. package/src/views/chat/components/chat-input.tsx +0 -298
  269. package/src/views/chat/components/header-thread-title.tsx +0 -210
  270. package/src/views/chat/components/message-list/content-answer.tsx +0 -100
  271. package/src/views/chat/components/message-list/content-question.tsx +0 -18
  272. package/src/views/chat/components/message-list/index.tsx +0 -520
  273. package/src/views/chat/components/message-list/message-item.tsx +0 -199
  274. package/src/views/chat/components/message-list/preparation-demo-items.ts +0 -147
  275. package/src/views/chat/components/message-list/preparation-steps.tsx +0 -506
  276. package/src/views/chat/components/message-list/suggestion-list.tsx +0 -36
  277. package/src/views/chat/components/message-list/thinking-process.tsx +0 -49
  278. package/src/views/chat/components/message-list/toolbar.tsx +0 -224
  279. package/src/views/chat/components/message-list/use-message-list-scroll.ts +0 -214
  280. package/src/views/chat/components/references-knowledge/context.tsx +0 -57
  281. package/src/views/chat/components/references-knowledge/index.ts +0 -9
  282. package/src/views/chat/components/references-knowledge/modal/knowledge-detail-drawer.tsx +0 -556
  283. package/src/views/chat/components/references-knowledge/modal/knowledge-doc-detail-drawer.tsx +0 -529
  284. package/src/views/chat/components/references-knowledge/panel.tsx +0 -115
  285. package/src/views/chat/hooks/use-chat-common.ts +0 -19
  286. package/src/views/chat/index-session.tsx +0 -647
  287. package/src/views/chat/index.tsx +0 -127
  288. package/src/views/page-error/401.tsx +0 -56
  289. package/src/views/page-error/404.tsx +0 -56
  290. package/src/views/page-menus/business-rules/index.tsx +0 -376
  291. package/src/views/page-menus/business-rules/modal/detail-modal.tsx +0 -186
  292. package/src/views/page-menus/business-rules/modal/scope-modal.tsx +0 -272
  293. package/src/views/page-menus/business-rules/modal/submit-modal.tsx +0 -142
  294. package/src/views/page-menus/feedback/components/feedback-dataset-list.tsx +0 -471
  295. package/src/views/page-menus/feedback/index.tsx +0 -166
  296. package/src/views/page-menus/feedback/modal/export-feedback-modal.tsx +0 -367
  297. package/src/views/page-menus/knowledge/components/doc-editor-by-type.tsx +0 -32
  298. package/src/views/page-menus/knowledge/components/doc-editor-type-file.tsx +0 -330
  299. package/src/views/page-menus/knowledge/detail.tsx +0 -600
  300. package/src/views/page-menus/knowledge/index.tsx +0 -337
  301. package/src/views/page-menus/knowledge/modal/detail-modal.tsx +0 -618
  302. package/src/views/page-menus/knowledge/modal/doc-detail-modal.tsx +0 -550
  303. package/src/views/page-menus/knowledge/modal/doc-parse.ts +0 -99
  304. package/src/views/page-menus/knowledge/modal/doc-submit-modal.tsx +0 -349
  305. package/src/views/page-menus/knowledge/modal/doc-type-picker-modal.tsx +0 -88
  306. package/src/views/page-menus/knowledge/modal/knowledge-user-select-modal.tsx +0 -283
  307. package/src/views/page-menus/knowledge/modal/submit-modal.tsx +0 -179
  308. package/src/views/page-menus/model-metadata/manage/components/metadata-detail-schema-tab.tsx +0 -114
  309. package/src/views/page-menus/model-metadata/manage/components/step1-basic-info.tsx +0 -232
  310. package/src/views/page-menus/model-metadata/manage/components/step2-schema.tsx +0 -316
  311. package/src/views/page-menus/model-metadata/manage/components/step3-permissions.tsx +0 -134
  312. package/src/views/page-menus/model-metadata/manage/components/step4-documents.tsx +0 -134
  313. package/src/views/page-menus/model-metadata/manage/components/step5-example-sql.tsx +0 -101
  314. package/src/views/page-menus/model-metadata/manage/components/submit-add.tsx +0 -338
  315. package/src/views/page-menus/model-metadata/manage/components/submit-edit.tsx +0 -276
  316. package/src/views/page-menus/model-metadata/manage/detail.tsx +0 -298
  317. package/src/views/page-menus/model-metadata/manage/hooks/model-metadata-submit-shared.ts +0 -113
  318. package/src/views/page-menus/model-metadata/manage/hooks/use-model-metadata-item-state.ts +0 -20
  319. package/src/views/page-menus/model-metadata/manage/index.tsx +0 -304
  320. package/src/views/page-menus/model-metadata/manage/modal/components/table-schema.ts +0 -374
  321. package/src/views/page-menus/model-metadata/manage/modal/components/use-table-detail-tabs.tsx +0 -151
  322. package/src/views/page-menus/model-metadata/manage/modal/components/use-table-submit-tabs.tsx +0 -423
  323. package/src/views/page-menus/model-metadata/manage/modal/detail-modal.tsx +0 -218
  324. package/src/views/page-menus/model-metadata/manage/modal/submit-modal.tsx +0 -261
  325. package/src/views/page-menus/model-metadata/manage/modal/table-detail-modal.tsx +0 -196
  326. package/src/views/page-menus/model-metadata/manage/modal/table-submit-modal.tsx +0 -229
  327. package/src/views/page-menus/model-metadata/manage/submit.tsx +0 -31
  328. package/src/views/page-menus/model-metadata/role/index.tsx +0 -207
  329. package/src/views/page-menus/model-metadata/role/modal/detail-modal.tsx +0 -97
  330. package/src/views/page-menus/model-metadata/role/modal/role-assign-users-modal.tsx +0 -254
  331. package/src/views/page-menus/model-metadata/role/modal/role-assign-users-panel.tsx +0 -393
  332. package/src/views/page-menus/model-metadata/role/modal/role-assign-users-utils.ts +0 -120
  333. package/src/views/page-menus/model-metadata/role/modal/role-permission-assign-panel.tsx +0 -698
  334. package/src/views/page-menus/model-metadata/role/modal/role-permission-modal.tsx +0 -237
  335. package/src/views/page-menus/model-metadata/role/modal/submit-modal.tsx +0 -135
  336. package/src/views/page-menus/model-metadata/training/components/detail-records/index.ts +0 -4
  337. package/src/views/page-menus/model-metadata/training/components/detail-records/node-card.tsx +0 -72
  338. package/src/views/page-menus/model-metadata/training/components/detail-records/summary-lines.ts +0 -196
  339. package/src/views/page-menus/model-metadata/training/components/detail-records/summary-list.tsx +0 -153
  340. package/src/views/page-menus/model-metadata/training/components/detail-records/timeline.tsx +0 -103
  341. package/src/views/page-menus/model-metadata/training/components/detail-records/types.ts +0 -82
  342. package/src/views/page-menus/model-metadata/training/detail.tsx +0 -159
  343. package/src/views/page-menus/model-metadata/training/index.tsx +0 -236
  344. package/src/views/page-menus/model-metadata/training/modal/update-detail-modal.tsx +0 -154
  345. package/src/views/page-menus/skill/index.tsx +0 -201
  346. package/src/views/page-menus/skill/modal/detail-modal.tsx +0 -156
  347. package/src/views/page-menus/skill/modal/submit-modal.tsx +0 -214
  348. package/src/views/page-system/agent-config/index.tsx +0 -370
  349. package/src/views/page-system/department/departmentFormShared.ts +0 -36
  350. package/src/views/page-system/department/index.tsx +0 -541
  351. package/src/views/page-system/department/modal/detail-modal.tsx +0 -94
  352. package/src/views/page-system/department/modal/member-role-modal.tsx +0 -128
  353. package/src/views/page-system/department/modal/submit-modal.tsx +0 -265
  354. package/src/views/page-system/dict/index.tsx +0 -440
  355. package/src/views/page-system/dict/modal/cate-submit-modal.tsx +0 -315
  356. package/src/views/page-system/dict/modal/submit-modal.tsx +0 -184
  357. package/src/views/page-system/logs/components/index.ts +0 -3
  358. package/src/views/page-system/logs/components/log-message-demo.tsx +0 -30
  359. package/src/views/page-system/logs/components/log-message-stream.ts +0 -184
  360. package/src/views/page-system/logs/components/message-list/content-answer.tsx +0 -100
  361. package/src/views/page-system/logs/components/message-list/content-question.tsx +0 -18
  362. package/src/views/page-system/logs/components/message-list/index.tsx +0 -515
  363. package/src/views/page-system/logs/components/message-list/message-item.tsx +0 -193
  364. package/src/views/page-system/logs/components/message-list/preparation-demo-items.ts +0 -147
  365. package/src/views/page-system/logs/components/message-list/preparation-steps.tsx +0 -506
  366. package/src/views/page-system/logs/components/message-list/suggestion-list.tsx +0 -36
  367. package/src/views/page-system/logs/components/message-list/thinking-process.tsx +0 -49
  368. package/src/views/page-system/logs/components/message-list/toolbar.tsx +0 -134
  369. package/src/views/page-system/logs/components/message-list/use-message-list-scroll.ts +0 -214
  370. package/src/views/page-system/logs/components/message-modal.tsx +0 -239
  371. package/src/views/page-system/logs/index.tsx +0 -132
  372. package/src/views/page-system/logs/modal/detail-modal.tsx +0 -157
  373. package/src/views/page-system/menu/components/menuFormShared.ts +0 -283
  374. package/src/views/page-system/menu/index.less +0 -12
  375. package/src/views/page-system/menu/index.tsx +0 -410
  376. package/src/views/page-system/menu/modal/icon-modal.less +0 -51
  377. package/src/views/page-system/menu/modal/icon-modal.tsx +0 -87
  378. package/src/views/page-system/menu/modal/submit-modal.tsx +0 -263
  379. package/src/views/page-system/permission/index.tsx +0 -562
  380. package/src/views/page-system/permission/modal/detail-modal.tsx +0 -179
  381. package/src/views/page-system/permission/modal/submit-modal.less +0 -146
  382. package/src/views/page-system/permission/modal/submit-modal.tsx +0 -650
  383. package/src/views/page-system/role/index.tsx +0 -163
  384. package/src/views/page-system/role/modal/detail-modal.tsx +0 -127
  385. package/src/views/page-system/role/modal/permission-assign-group-rules.ts +0 -86
  386. package/src/views/page-system/role/modal/permission-modal.tsx +0 -111
  387. package/src/views/page-system/role/modal/role-modal-shell-styles.ts +0 -21
  388. package/src/views/page-system/role/modal/role-permission-assign-panel.tsx +0 -916
  389. package/src/views/page-system/role/modal/role-permission-assign-shared.ts +0 -1047
  390. package/src/views/page-system/role/modal/submit-modal.tsx +0 -193
  391. package/src/views/page-system/sys-config/index.tsx +0 -294
  392. package/src/views/page-system/user/components/user-role-column.tsx +0 -87
  393. package/src/views/page-system/user/index.tsx +0 -439
  394. package/src/views/page-system/user/modal/assign-roles-modal.tsx +0 -389
  395. package/src/views/page-system/user/modal/detail-modal.tsx +0 -72
  396. package/src/views/page-system/user/modal/modal-style/submit-modal.less +0 -40
  397. package/src/views/page-system/user/modal/submit-modal.less +0 -40
  398. package/src/views/page-system/user/modal/submit-modal.tsx +0 -287
  399. package/src/views/page-system/user/userFormShared.ts +0 -51
  400. package/tailwind.config.js +0 -17
  401. package/tsconfig.app.json +0 -48
  402. package/tsconfig.json +0 -11
  403. package/tsconfig.node.json +0 -26
  404. package/vite.config.ts +0 -272
  405. /package/{public → dist}/favicon.ico +0 -0
  406. /package/{public → dist}/vite.svg +0 -0
@@ -1,1049 +0,0 @@
1
- import { Form } from 'antd'
2
- import React, {
3
- useCallback,
4
- useEffect,
5
- useLayoutEffect,
6
- useMemo,
7
- useRef,
8
- useState,
9
- } from 'react'
10
- import { flushSync } from 'react-dom'
11
-
12
- import ConfirmDialog from '@/components/common/modal/confirm-dialog'
13
- import Pagination from '@/components/common/pagination'
14
- import { useAuth } from '@/contexts/AuthContext'
15
- import { useMenu } from '@/contexts/MenuContext'
16
- import { useRouter } from '@/hooks/useRouter'
17
- import { useStickyHeader } from '@/hooks/useStickyHeader'
18
- import { ChevronLeft, Plus, ShieldUser } from 'lucide-react'
19
- import { useTranslation } from 'react-i18next'
20
- import { useLocation, useNavigate } from 'react-router-dom'
21
-
22
- import { CxButton, CxEmptyState, CxTable, CxTooltip } from '@cx-ui'
23
- import { getListPathFromSubRoutePathname } from '@/utils/menuRouteRegistry'
24
- import { flattenMenuTree, getIconComponent } from '@/utils/routeConfig'
25
- import { SearchArea } from './components/search-area'
26
- import {
27
- ListPageViewProps,
28
- ListRenderContext,
29
- SearchButtonSize,
30
- TABLE_VIEW_DEFAULT_PAGE_SIZE_OPTIONS,
31
- TABLE_VIEW_TREE_DEFAULTS,
32
- TreeConfig,
33
- TreeContext,
34
- } from './components/table-view.types'
35
- import { normalizeTreeAPIResult } from './components/tree-api-normalize'
36
- import { annotateTreeDisableNode } from './components/tree-data-annotate'
37
- import { TreeSidebar } from './components/tree-sidebar'
38
-
39
- const isDev = import.meta.env.DEV
40
-
41
- function collectAllTreeNodeKeys(
42
- nodes: any[] | undefined,
43
- idKey: string,
44
- childrenKey: string,
45
- ): (string | number)[] {
46
- const keys: (string | number)[] = []
47
- const walk = (list: any[]) => {
48
- for (const n of list) {
49
- const id = n?.[idKey]
50
- if (id !== undefined && id !== null) keys.push(id)
51
- const children = n?.[childrenKey]
52
- if (Array.isArray(children) && children.length) walk(children)
53
- }
54
- }
55
- if (Array.isArray(nodes)) walk(nodes)
56
- return keys
57
- }
58
-
59
- function normalizeSearchValuesByFields(
60
- values: Record<string, any>,
61
- searchFields: ListPageViewProps['pageConfig']['searchFields'],
62
- ) {
63
- const next: Record<string, any> = { ...values }
64
- const fields = searchFields ?? []
65
- for (const field of fields) {
66
- if (field.type !== 'dateRange') continue
67
-
68
- const startKey = field.rangeStartProp || `${field.prop}Start`
69
- const endKey = field.rangeEndProp || `${field.prop}End`
70
- const format = field.format || 'YYYY-MM-DD HH:mm:ss'
71
- const range = values[field.prop]
72
-
73
- if (Array.isArray(range) && range[0] && range[1]) {
74
- if (field.showTime) {
75
- next[startKey] = range[0].format(format)
76
- next[endKey] = range[1].format(format)
77
- } else {
78
- next[startKey] = range[0].startOf('day').format(format)
79
- next[endKey] = range[1].endOf('day').format(format)
80
- }
81
- } else {
82
- next[startKey] = undefined
83
- next[endKey] = undefined
84
- }
85
-
86
- }
87
- return next
88
- }
89
-
90
- /** 删除确认文案 `t(content, …)` 的插值对象;未配 `contentFields` 时默认只取 `name` */
91
- function buildDeleteConfirmInterpolation(
92
- record: any | null,
93
- contentFields?: string | string[],
94
- ): Record<string, unknown> {
95
- if (record == null) return { name: '' }
96
-
97
- let keys: string[]
98
- if (contentFields == null || contentFields === '') {
99
- keys = ['name']
100
- } else if (Array.isArray(contentFields)) {
101
- keys = contentFields.filter(
102
- (x): x is string => typeof x === 'string' && x.length > 0,
103
- )
104
- if (keys.length === 0) keys = ['name']
105
- } else {
106
- keys = [contentFields]
107
- }
108
-
109
- const out: Record<string, unknown> = {}
110
- for (const k of keys) {
111
- out[k] = record[k]
112
- }
113
- return out
114
- }
115
-
116
- type TableActionsRender = (ctx: {
117
- searchButtonSize: SearchButtonSize
118
- tree?: TreeContext
119
- }) => React.ReactNode
120
- type HeaderActionsRender = (ctx: { tree?: TreeContext }) => React.ReactNode
121
-
122
- function TableActions({
123
- children,
124
- searchButtonSize,
125
- tree,
126
- }: {
127
- children?: React.ReactNode | TableActionsRender
128
- searchButtonSize: SearchButtonSize
129
- tree?: TreeContext
130
- }) {
131
- if (children == null) return null
132
-
133
- const content =
134
- typeof children === 'function'
135
- ? (children as TableActionsRender)({ tree, searchButtonSize })
136
- : children
137
- if (content == null || content === false) return null
138
-
139
- return (
140
- <div className="px-3 py-4 border-b border-gray-200 dark:border-gray-700 flex justify-between items-start">
141
- {content}
142
- </div>
143
- )
144
- }
145
-
146
- /** 根 props 与 pageConfig 解构字段按字母序,便于与 `table-view.types` 对照。 */
147
- function ListPageView({
148
- columns,
149
- detailModalComponent,
150
- headerActions,
151
- pageConfig,
152
- renderList,
153
- searchAreaExtra,
154
- searchSlot,
155
- showBackToList,
156
- submitModalComponent,
157
- tableActions,
158
- titleExtra,
159
- titleIcon,
160
- }: ListPageViewProps) {
161
- const { t } = useTranslation()
162
- const navigate = useNavigate()
163
- const location = useLocation()
164
- const router = useRouter()
165
- const { menus } = useMenu()
166
- const { hasPermission } = useAuth()
167
- const SubmitModalComp = submitModalComponent as React.ElementType | undefined
168
- const DetailModalComp = detailModalComponent as React.ElementType | undefined
169
-
170
- const {
171
- afterStatusChange,
172
- deleteConfirm,
173
- detailAction,
174
- getDetailModalOpenExtra,
175
- fetchTableData,
176
- handleDelete: rawHandleDelete,
177
- handleUpdateStatus: rawHandleUpdateStatus,
178
- handleReset,
179
- handleSearch,
180
- handleTableChange,
181
- loading,
182
- onAdd,
183
- onDetail,
184
- onEdit,
185
- pageInfo = {
186
- listTitle: undefined,
187
- subTitle: undefined,
188
- title: undefined,
189
- },
190
- pageSizeOptions: pageSizeOptionsFromConfig,
191
- paginationData,
192
- patchSearchAndRefresh,
193
- replaceSearchAndRefresh,
194
- searchFields,
195
- searchFormData,
196
- setSearchFormData,
197
- statusToggle,
198
- submitAction,
199
- table: tableConfig = {},
200
- tableData,
201
- tree,
202
- treeAPI,
203
- treeReloadSignal = 0,
204
- } = pageConfig
205
-
206
- const searchButtonSize: SearchButtonSize =
207
- pageInfo.searchButtonSize ?? 'default'
208
-
209
- const pageSizeOptions =
210
- pageSizeOptionsFromConfig?.length
211
- ? pageSizeOptionsFromConfig
212
- : [...TABLE_VIEW_DEFAULT_PAGE_SIZE_OPTIONS]
213
-
214
- const table = tableConfig || {}
215
- const tableScrollX =
216
- table.scrollX === false ? undefined : ((table.scrollX ?? 'max-content') as any)
217
- const isSlicePagination = table.paginationMode === 'slice'
218
-
219
- const hasTree = !!(treeAPI || tree)
220
- const mergedTree: TreeConfig | undefined = useMemo(() => {
221
- if (!hasTree) return undefined
222
- return {
223
- ...TABLE_VIEW_TREE_DEFAULTS,
224
- ...tree,
225
- title: tree?.title ?? pageInfo.title,
226
- }
227
- }, [hasTree, tree, pageInfo.title])
228
-
229
- const onTreeDataChangeRef = useRef(mergedTree?.onTreeDataChange)
230
- onTreeDataChangeRef.current = mergedTree?.onTreeDataChange
231
-
232
- const stickyTitle = pageInfo.title
233
- ? t(String(pageInfo.title))
234
- : t('common.list_page_title', { defaultValue: 'List' })
235
- const titleRef = useStickyHeader(stickyTitle)
236
-
237
- // 删除确认弹框状态
238
- const [deleteState, setDeleteState] = useState<{
239
- open: boolean
240
- record: any | null
241
- }>({
242
- open: false,
243
- record: null,
244
- })
245
-
246
- const openDeleteConfirm = useCallback((record: any) => {
247
- setDeleteState({
248
- open: true,
249
- record,
250
- })
251
- }, [])
252
-
253
- const closeDeleteConfirm = useCallback(() => {
254
- setDeleteState({
255
- open: false,
256
- record: null,
257
- })
258
- }, [])
259
-
260
- const [statusToggleState, setStatusToggleState] = useState<{
261
- nextStatus: number | null
262
- open: boolean
263
- record: any | null
264
- }>({
265
- nextStatus: null,
266
- open: false,
267
- record: null,
268
- })
269
-
270
- const openStatusToggleConfirm = useCallback(
271
- (payload: { nextStatus: number; data: any }) => {
272
- setStatusToggleState({
273
- nextStatus: payload.nextStatus,
274
- open: true,
275
- record: payload.data,
276
- })
277
- },
278
- [],
279
- )
280
-
281
- const closeStatusToggleConfirm = useCallback(() => {
282
- setStatusToggleState({
283
- nextStatus: null,
284
- open: false,
285
- record: null,
286
- })
287
- }, [])
288
-
289
- const activeStatusToggleConfirm = useMemo(() => {
290
- const st = statusToggle ?? {}
291
- const enabledStatus = st.enabledStatus ?? 1
292
- if (statusToggleState.nextStatus == null) return undefined
293
- return statusToggleState.nextStatus === enabledStatus
294
- ? st.enableConfirm
295
- : st.disableConfirm
296
- }, [statusToggle, statusToggleState.nextStatus])
297
-
298
- const statusToggleContentInterpolation = useMemo(
299
- () =>
300
- buildDeleteConfirmInterpolation(
301
- statusToggleState.record,
302
- activeStatusToggleConfirm?.contentFields,
303
- ),
304
- [activeStatusToggleConfirm?.contentFields, statusToggleState.record],
305
- )
306
-
307
- const statusToggleConfirmDanger = useMemo(() => {
308
- const st = statusToggle ?? {}
309
- const enabledStatus = st.enabledStatus ?? 1
310
- if (statusToggleState.nextStatus == null) return false
311
- const isDisable = statusToggleState.nextStatus !== enabledStatus
312
- if (!isDisable) return false
313
- return st.dangerOnDisable !== false
314
- }, [statusToggle, statusToggleState.nextStatus])
315
-
316
- const deleteContentInterpolation = useMemo(
317
- () =>
318
- buildDeleteConfirmInterpolation(
319
- deleteState.record,
320
- deleteConfirm?.contentFields,
321
- ),
322
- [deleteState.record, deleteConfirm?.contentFields],
323
- )
324
-
325
- const handleConfirmDelete = async () => {
326
- if (!deleteState.record) return
327
- const targetRecord = deleteState.record
328
- if (typeof rawHandleDelete !== 'function') {
329
- console.error('[ListPageView] handleDelete 未注入,无法执行删除请求')
330
- closeDeleteConfirm()
331
- return
332
- }
333
- const deleted = await rawHandleDelete(targetRecord)
334
- if (deleted !== false && typeof pageConfig.afterDelete === 'function') {
335
- await pageConfig.afterDelete(targetRecord)
336
- }
337
- closeDeleteConfirm()
338
- }
339
-
340
- const handleConfirmStatusToggle = async () => {
341
- if (
342
- statusToggleState.record == null ||
343
- statusToggleState.nextStatus == null
344
- ) {
345
- return
346
- }
347
- const targetRecord = statusToggleState.record
348
- const nextStatus = statusToggleState.nextStatus
349
- if (typeof rawHandleUpdateStatus !== 'function') {
350
- console.error(
351
- '[ListPageView] handleUpdateStatus 未注入,无法执行状态更新请求',
352
- )
353
- closeStatusToggleConfirm()
354
- return
355
- }
356
- const ok = await rawHandleUpdateStatus(targetRecord, nextStatus)
357
- if (ok !== false && typeof afterStatusChange === 'function') {
358
- await afterStatusChange(targetRecord, nextStatus)
359
- }
360
- closeStatusToggleConfirm()
361
- }
362
-
363
- // 提交弹框 ref(新增 / 编辑)
364
- const submitModalRef = React.useRef<any>(null)
365
-
366
- // 详情弹框 ref
367
- const detailModalRef = React.useRef<any>(null)
368
-
369
- const openEditDefault = useCallback(
370
- (record: any) => {
371
- const config = submitAction
372
-
373
- if (config?.submitType === 'modal' && SubmitModalComp) {
374
- submitModalRef.current?.open({
375
- opeType: 'edit',
376
- opeId: record?.id,
377
- data: record,
378
- })
379
- return
380
- }
381
-
382
- if (config?.submitType === 'page') {
383
- const id = record?.id
384
- if (id) {
385
- // 编辑模式,携带 ID
386
- router.toEdit({
387
- state: { editId: id },
388
- })
389
- } else {
390
- // 新增模式
391
- router.toAdd()
392
- }
393
- return
394
- }
395
-
396
- if (isDev) {
397
- console.log(
398
- '默认编辑处理(请在 submitAction 中配置 submitType 或在 onEdit 中重写)',
399
- { record, config },
400
- )
401
- }
402
- },
403
- [submitAction, SubmitModalComp],
404
- )
405
-
406
- const getDetailModalOpenExtraRef = useRef(getDetailModalOpenExtra)
407
- getDetailModalOpenExtraRef.current = getDetailModalOpenExtra
408
-
409
- const openDetailDefault = useCallback(
410
- (record: any) => {
411
- const config = detailAction
412
- const id = record?.id
413
-
414
- if (config?.detailType === 'modal' && DetailModalComp) {
415
- const extra =
416
- typeof getDetailModalOpenExtraRef.current === 'function'
417
- ? getDetailModalOpenExtraRef.current() ?? {}
418
- : {}
419
- detailModalRef.current?.open({
420
- opeType: 'detail',
421
- opeId: id,
422
- data: record,
423
- ...extra,
424
- })
425
- return
426
- }
427
-
428
- if (config?.detailType === 'page') {
429
- const id = record?.id || record?.code
430
- if (id) {
431
- router.toDetail(id)
432
- }
433
- return
434
- }
435
-
436
- if (isDev) {
437
- console.log(
438
- '默认详情处理(请在 detailAction 中配置 detailType 或在 onDetail 中重写)',
439
- { record, config },
440
- )
441
- }
442
- },
443
- [detailAction, DetailModalComp],
444
- )
445
-
446
- useLayoutEffect(() => {
447
- const page = pageConfig
448
- page.onDelete = openDeleteConfirm
449
- page.onEdit = (payload: {
450
- config?: any
451
- data?: any
452
- type: string
453
- }) => {
454
- if (onEdit) {
455
- onEdit(payload)
456
- return
457
- }
458
- openEditDefault(payload?.data)
459
- }
460
- page.onDetail = (payload: {
461
- config?: any
462
- data?: any
463
- type: string
464
- }) => {
465
- if (onDetail) {
466
- onDetail(payload)
467
- return
468
- }
469
- openDetailDefault(payload?.data)
470
- }
471
- page.onStatusToggle = openStatusToggleConfirm
472
- return () => {
473
- if (page.onDelete === openDeleteConfirm) {
474
- delete page.onDelete
475
- }
476
- delete page.onEdit
477
- delete page.onDetail
478
- if (page.onStatusToggle === openStatusToggleConfirm) {
479
- delete page.onStatusToggle
480
- }
481
- }
482
- }, [
483
- pageConfig,
484
- openDeleteConfirm,
485
- openStatusToggleConfirm,
486
- onEdit,
487
- onDetail,
488
- openEditDefault,
489
- openDetailDefault,
490
- ])
491
-
492
- // 新增
493
- const handleAdd = () => {
494
- const config = submitAction
495
-
496
- if (onAdd) {
497
- onAdd({ type: 'add', config })
498
- return
499
- }
500
-
501
- if (config?.submitType === 'modal' && SubmitModalComp) {
502
- submitModalRef.current?.open({ opeType: 'add' })
503
- return
504
- }
505
-
506
- if (config?.submitType === 'page') {
507
- router.toAdd()
508
- return
509
- }
510
-
511
- if (isDev) {
512
- console.log(
513
- '默认新增处理(请在 submitAction 中配置 submitType 或在 onAdd 中重写)',
514
- config,
515
- )
516
- }
517
- }
518
-
519
- const renderCell = (col: any, record: any, rowIndex: number) => {
520
- const fieldKey = col.dataIndex ?? col.key
521
- if (col.render) {
522
- return col.render(record[fieldKey], record, rowIndex)
523
- }
524
- const raw = record[fieldKey]
525
- if (col.ellipsis) {
526
- const s = raw != null && raw !== '' ? String(raw) : '-'
527
- const inner = (
528
- <div className="min-w-0 max-w-full overflow-hidden">
529
- <span className="block truncate">{s}</span>
530
- </div>
531
- )
532
- if (s === '-') return inner
533
- const placement = col.ellipsisTooltipPlacement ?? 'right'
534
- const tw = col.ellipsisTooltipWidth
535
- const maxW =
536
- tw == null || tw === ''
537
- ? '300px'
538
- : typeof tw === 'number'
539
- ? `${tw}px`
540
- : String(tw)
541
- return (
542
- <CxTooltip
543
- placement={placement}
544
- styles={{
545
- root: { maxWidth: maxW },
546
- }}
547
- title={
548
- <div
549
- className="max-h-[50vh] overflow-y-auto whitespace-pre-wrap break-words text-left"
550
- style={{ maxWidth: maxW }}
551
- >
552
- {s}
553
- </div>
554
- }
555
- >
556
- {inner}
557
- </CxTooltip>
558
- )
559
- }
560
- return raw
561
- }
562
-
563
- const [form] = Form.useForm()
564
-
565
- const applySearchFromForm = () => {
566
- const vals = form.getFieldsValue(true)
567
- let normalizedVals = normalizeSearchValuesByFields(vals, searchFields)
568
- const treeListSearchKey =
569
- mergedTree?.searchField ??
570
- (treeAPI ? TABLE_VIEW_TREE_DEFAULTS.searchField : undefined)
571
- if (
572
- treeListSearchKey &&
573
- searchFormData &&
574
- Object.prototype.hasOwnProperty.call(searchFormData, treeListSearchKey)
575
- ) {
576
- normalizedVals = {
577
- ...normalizedVals,
578
- [treeListSearchKey]: searchFormData[treeListSearchKey],
579
- }
580
- }
581
- flushSync(() => {
582
- setSearchFormData?.(normalizedVals)
583
- })
584
- handleSearch()
585
- }
586
-
587
- const onReset = () => {
588
- handleReset()
589
- }
590
-
591
- const submitPermission = submitAction?.permission ?? ''
592
- const primaryLabelKey = submitAction?.labelKey ?? 'p_system.role.add'
593
-
594
- // ===== Tree 状态 & 渲染 =====
595
- const [treeData, setTreeData] = useState<any[]>([])
596
- const [treeDataLoaded, setTreeDataLoaded] = useState(!hasTree)
597
- const [expandedKeys, setExpandedKeys] = useState<Set<string | number>>(
598
- new Set(),
599
- )
600
- const [selectedKey, setSelectedKey] = useState<string | number | null>(null)
601
-
602
- const idKey = mergedTree?.fieldNames?.id ?? 'id'
603
- const childrenKey = mergedTree?.fieldNames?.children ?? 'children'
604
- const treeDataForSidebar = useMemo(() => {
605
- const disableNode = mergedTree?.disableNode
606
- if (!disableNode || !treeData.length) return treeData
607
- return annotateTreeDisableNode(
608
- treeData,
609
- idKey,
610
- childrenKey,
611
- disableNode,
612
- )
613
- }, [treeData, mergedTree?.disableNode, idKey, childrenKey])
614
- const searchField = mergedTree?.searchField ?? 'id'
615
- const autoSearch = mergedTree?.autoSearch ?? true
616
- const unselectText = mergedTree?.unselectText ?? 'common.unselect_tree'
617
-
618
- useEffect(() => {
619
- if (!treeAPI) {
620
- setTreeDataLoaded(true)
621
- if (hasTree && isDev) {
622
- console.warn('[ListPageView] 已启用树区域但未提供 treeAPI')
623
- }
624
- return
625
- }
626
-
627
- let mounted = true
628
- setTreeDataLoaded(false)
629
-
630
- treeAPI()
631
- .then((raw) => {
632
- if (!mounted) return
633
- const normalized = normalizeTreeAPIResult(raw)
634
- setTreeData(normalized)
635
- setTreeDataLoaded(true)
636
- onTreeDataChangeRef.current?.(normalized)
637
- })
638
- .catch((error) => {
639
- if (!mounted) return
640
- console.error('[ListPageView] Tree fetch error:', error)
641
- setTreeDataLoaded(true)
642
- })
643
- return () => {
644
- mounted = false
645
- }
646
- }, [treeAPI, hasTree, treeReloadSignal])
647
-
648
- useEffect(() => {
649
- if (!mergedTree?.defaultExpandAll || !treeData.length) return
650
- setExpandedKeys(
651
- new Set(collectAllTreeNodeKeys(treeData, idKey, childrenKey)),
652
- )
653
- }, [treeData, mergedTree?.defaultExpandAll, idKey, childrenKey])
654
-
655
- const findNodeByKey = (nodes: any[], key: any): any | null => {
656
- for (const n of nodes) {
657
- if (n[idKey] === key) return n
658
- const children = n[childrenKey]
659
- if (Array.isArray(children)) {
660
- const found = findNodeByKey(children, key)
661
- if (found) return found
662
- }
663
- }
664
- return null
665
- }
666
-
667
- const toggleTreeExpand = (key: any, e?: React.MouseEvent) => {
668
- if (e) e.stopPropagation()
669
- setExpandedKeys((prev) => {
670
- const next = new Set(prev)
671
- if (next.has(key)) next.delete(key)
672
- else next.add(key)
673
- return next
674
- })
675
- }
676
-
677
- const handleTreeSelect = (key: any) => {
678
- const formSnap = normalizeSearchValuesByFields(
679
- form.getFieldsValue(true),
680
- searchFields ?? [],
681
- )
682
- const mergedBase = { ...(searchFormData || {}), ...formSnap }
683
-
684
- if (key == null) {
685
- setSelectedKey(null)
686
- mergedTree?.onSelect?.(null)
687
- if (!mergedTree?.searchField) return
688
- const next = { ...mergedBase }
689
- delete next[searchField]
690
- if (replaceSearchAndRefresh && autoSearch) {
691
- replaceSearchAndRefresh(next)
692
- } else if (setSearchFormData) {
693
- flushSync(() => {
694
- setSearchFormData(next)
695
- })
696
- if (autoSearch) handleSearch()
697
- }
698
- return
699
- }
700
-
701
- setSelectedKey(key)
702
- const node = findNodeByKey(treeData, key)
703
- if (!node) return
704
-
705
- if (mergedTree?.onSelect) {
706
- mergedTree.onSelect(node)
707
- }
708
-
709
- if (!mergedTree?.searchField) return
710
-
711
- const full = { ...mergedBase, [searchField]: node[idKey] }
712
-
713
- if (patchSearchAndRefresh && autoSearch) {
714
- patchSearchAndRefresh(full)
715
- } else if (setSearchFormData) {
716
- flushSync(() => {
717
- setSearchFormData(full)
718
- })
719
- if (autoSearch) handleSearch()
720
- }
721
- }
722
-
723
- // 默认 defaultSelectRoot 为 true;仅显式 false 时不自动选根
724
- useEffect(() => {
725
- if (mergedTree?.defaultSelectRoot === false) return
726
- if (!treeDataLoaded || treeData.length === 0) return
727
- if (selectedKey != null) return
728
- const root = treeData[0]
729
- const key = root?.[idKey]
730
- if (key === undefined || key === null) return
731
- handleTreeSelect(key)
732
- // eslint-disable-next-line react-hooks/exhaustive-deps -- 仅在树就绪且无选中时触发;避免 handleTreeSelect 变化导致重复
733
- }, [
734
- mergedTree?.defaultSelectRoot,
735
- treeDataLoaded,
736
- treeData,
737
- idKey,
738
- selectedKey,
739
- ])
740
-
741
- const selectedNode =
742
- selectedKey != null ? findNodeByKey(treeData, selectedKey) : null
743
- const treeCtx: TreeContext | undefined = mergedTree
744
- ? { selectedKey, selectedNode, treeData, treeDataLoaded }
745
- : undefined
746
-
747
- const listRenderContext: ListRenderContext & {
748
- tree?: TreeContext
749
- } = {
750
- columns: columns ?? [],
751
- renderCell,
752
- tableData,
753
- t,
754
- tree: treeCtx,
755
- }
756
-
757
- // 将 treeContext 暴露给 pageConfig,供页面组件使用
758
- React.useEffect(() => {
759
- if (pageConfig) {
760
- ;(pageConfig as any).treeContext = treeCtx
761
- }
762
- }, [treeCtx, pageConfig])
763
-
764
- const showListArea =
765
- !mergedTree ||
766
- !!selectedKey ||
767
- mergedTree?.sameNodeReselect === 'deselect'
768
- const showNoDataEmpty = showListArea &&
769
- (isSlicePagination ? tableData.length === 0 : paginationData.total === 0)
770
-
771
- // 分页与表格边线处理交给 CSS(分页区 -1px 覆盖最后一行 border),避免监听滚动带来的卡顿
772
- const headerActionsContent =
773
- typeof headerActions === 'function'
774
- ? (headerActions as HeaderActionsRender)({ tree: treeCtx })
775
- : headerActions
776
- const hasHeaderActionsContent =
777
- headerActionsContent !== undefined &&
778
- headerActionsContent !== null &&
779
- headerActionsContent !== false
780
-
781
- const searchAreaExtraNode =
782
- searchAreaExtra == null || searchAreaExtra === false
783
- ? null
784
- : typeof searchAreaExtra === 'function'
785
- ? (
786
- searchAreaExtra as (ctx: {
787
- searchButtonSize: SearchButtonSize
788
- tree?: TreeContext
789
- }) => React.ReactNode
790
- )({ tree: treeCtx, searchButtonSize })
791
- : searchAreaExtra
792
-
793
- /** 顶部标题图标:未显式传入时从菜单配置自动推导。 */
794
- const resolvedTitleIcon = useMemo(() => {
795
- if (titleIcon) return titleIcon
796
- const currentPath = location.pathname || ''
797
- const listPath = getListPathFromSubRoutePathname(currentPath) || currentPath
798
- const normalize = (p: string) =>
799
- (p || '')
800
- .replace(/\/(index|detail|submit)(\/.*)?$/, '')
801
- .replace(/\/+$/, '')
802
- const target = normalize(listPath)
803
- if (!target) return undefined
804
-
805
- const flat = flattenMenuTree(menus || [])
806
- const matched = flat.find((m) => normalize(m?.path || '') === target)
807
- const Icon = getIconComponent(matched?.icon)
808
- if (!Icon) return undefined
809
- return <Icon className="text-blue-400" size={32} />
810
- }, [location.pathname, menus, titleIcon])
811
-
812
- // ===== 统一外层布局:顶部标题区域 + 中间内容(可选左树右表) =====
813
- return (
814
- <div className="p-8 mx-auto w-full flex-1 h-full min-h-0 flex flex-col box-border overflow-hidden">
815
- {/* 顶部标题区域 */}
816
- <div className="flex justify-between items-center mb-6">
817
- <div
818
- className={`flex min-w-0 gap-3 ${showBackToList ? 'items-center' : 'items-start gap-2'}`}>
819
- {showBackToList ? (
820
- <CxButton.Icon
821
- className="!p-2.5 shrink-0 rounded-lg border border-solid border-gray-300 bg-white shadow-sm dark:border-gray-600 dark:bg-gray-900"
822
- icon={<ChevronLeft size={22} strokeWidth={2.25} />}
823
- title={t('common.back')}
824
- tooltipTone="default"
825
- type="default"
826
- onClick={() => router.toIndex()}
827
- />
828
- ) : null}
829
- <div className="min-w-0 flex-1">
830
- <h1
831
- ref={titleRef}
832
- className="text-3xl font-bold bg-clip-text text-transparent bg-gradient-to-r from-blue-400 to-purple-500 flex flex-wrap items-center gap-3">
833
- {resolvedTitleIcon ?? <ShieldUser className="text-blue-400" size={32} />}
834
- {pageInfo.title ? t(String(pageInfo.title)) : ''}
835
- {titleExtra}
836
- </h1>
837
- <p className="text-gray-500 dark:text-gray-400 mt-1">
838
- {pageInfo.subTitle ? t(String(pageInfo.subTitle)) : ''}
839
- </p>
840
- </div>
841
- </div>
842
- <div className="flex flex-wrap gap-2 items-center justify-end">
843
- {headerActionsContent}
844
- {!hasHeaderActionsContent && submitPermission && hasPermission(submitPermission) && (
845
- <CxButton.Add
846
- size="medium"
847
- icon={<Plus size={20} />}
848
- onClick={handleAdd}>
849
- {t(primaryLabelKey)}
850
- </CxButton.Add>
851
- )}
852
- </div>
853
- </div>
854
- {/* 中间区域:有树时左右分栏,否则只显示右侧卡片 */}
855
- <div
856
- className="flex flex-1 min-h-0 gap-6"
857
- style={{ minHeight: 0 }}>
858
- {/* 左侧树:数据来自 treeAPI;tree 仅合并 TreeConfig(字段名、defaultSelectRoot 等) */}
859
- {mergedTree && (
860
- <TreeSidebar
861
- expandedKeys={expandedKeys}
862
- onSelect={handleTreeSelect}
863
- onToggleExpand={toggleTreeExpand}
864
- selectedKey={selectedKey}
865
- tree={mergedTree}
866
- treeData={treeDataForSidebar}
867
- />
868
- )}
869
- {/* 右侧:Header / Toolbar + 搜索 + 表格 + 分页,统一包在一张大卡片里 */}
870
- <div
871
- className="flex-1 min-w-0 flex flex-col"
872
- >
873
- <div className="flex-1 bg-white dark:bg-gray-800 rounded-xl shadow-sm border border-gray-200 dark:border-gray-700 flex flex-col overflow-hidden">
874
- {/* Header / Toolbar(可访问树上下文) */}
875
- {tableActions &&
876
- (!mergedTree ||
877
- selectedKey != null ||
878
- mergedTree?.sameNodeReselect === 'deselect') ? (
879
- <TableActions
880
- searchButtonSize={searchButtonSize}
881
- tree={treeCtx}>
882
- {tableActions}
883
- </TableActions>
884
- ) : null}
885
- {/* 搜索区域 */}
886
- {searchSlot ? (
887
- searchSlot
888
- ) : ((mergedTree &&
889
- (selectedKey != null ||
890
- mergedTree?.sameNodeReselect === 'deselect')) ||
891
- !mergedTree) &&
892
- searchFields &&
893
- searchFields.length > 0 ? (
894
- <SearchArea
895
- extra={searchAreaExtraNode}
896
- form={form}
897
- listTitle={pageInfo.listTitle}
898
- onReset={onReset}
899
- onSearch={applySearchFromForm}
900
- searchButtonSize={searchButtonSize}
901
- searchFields={searchFields}
902
- searchFormData={searchFormData}
903
- total={paginationData.total}
904
- />
905
- ) : null}
906
- {/* 表格区域(表头固定:由 Table 内部滚动承载) */}
907
- <div className="flex min-h-0 flex-1 flex-col overflow-hidden">
908
- {showListArea && (!showNoDataEmpty || renderList) && (
909
- <div className="min-h-0 flex-1">
910
- {renderList ? (
911
- renderList(listRenderContext)
912
- ) : (
913
- <div className="h-full min-h-0">
914
- <CxTable
915
- autoScrollY
916
- minScrollY={160}
917
- sticky
918
- dataSource={tableData}
919
- rowKey="id"
920
- // 透传列配置,避免丢失 minWidth/ellipsis/fixed/onCell 等属性
921
- columns={(columns ?? []).map((col: any) => ({
922
- ...col,
923
- key: col.key ?? col.dataIndex,
924
- dataIndex: col.dataIndex ?? col.key,
925
- title:
926
- typeof col.title === 'string' ? t(col.title) : col.title,
927
- render: (_: unknown, record: any, rowIndex: number) =>
928
- renderCell(col, record, rowIndex),
929
- }))}
930
- tableLayout={tableScrollX ? 'auto' : 'fixed'}
931
- scroll={tableScrollX ? ({ x: tableScrollX } as any) : undefined}
932
- />
933
- </div>
934
- )}
935
- </div>
936
- )}
937
-
938
- {mergedTree &&
939
- selectedKey == null &&
940
- mergedTree?.sameNodeReselect !== 'deselect' && (
941
- <CxEmptyState message={t(unselectText)} />
942
- )}
943
-
944
- {!renderList && showNoDataEmpty && <CxEmptyState message={t('common.no_data')} />}
945
- </div>
946
- {/* 分页 */}
947
- {table.showPagination !== false &&
948
- (isSlicePagination || paginationData.total > 0) && (
949
- <div className="relative z-[1] -mt-px px-3 py-3 bg-white dark:bg-gray-800 border-t border-gray-200 dark:border-gray-700">
950
- <Pagination
951
- onPageChange={(page) => {
952
- handleTableChange({
953
- current: page,
954
- pageSize: paginationData.pageSize,
955
- })
956
- }}
957
- onPageSizeChange={(size) => {
958
- handleTableChange({
959
- current: paginationData.page,
960
- pageSize: size,
961
- })
962
- }}
963
- page={paginationData.page}
964
- pageSize={paginationData.pageSize}
965
- pageSizeOptions={pageSizeOptions}
966
- showSizeChanger
967
- showStats={!isSlicePagination}
968
- variant={isSlicePagination ? 'slice' : 'spread'}
969
- total={paginationData.total}
970
- isFirst={paginationData.first}
971
- isLast={paginationData.last}
972
- zeroIndexed={false}
973
- disabled={loading}
974
- />
975
- </div>
976
- )}
977
- </div>
978
- </div>
979
- </div>
980
-
981
- {/* 删除确认弹窗 & 提交 / 详情弹窗 */}
982
- <ConfirmDialog
983
- cancelText={
984
- deleteConfirm?.cancelText ? String(t(deleteConfirm.cancelText)) : undefined
985
- }
986
- confirmText={
987
- deleteConfirm?.okText ? String(t(deleteConfirm.okText)) : undefined
988
- }
989
- danger
990
- isOpen={deleteState.open}
991
- message={
992
- deleteConfirm?.content
993
- ? String(t(deleteConfirm.content, deleteContentInterpolation))
994
- : ''
995
- }
996
- title={String(t(deleteConfirm?.title ?? 'common.tip'))}
997
- onClose={closeDeleteConfirm}
998
- onConfirm={handleConfirmDelete}
999
- />
1000
-
1001
- <ConfirmDialog
1002
- cancelText={
1003
- activeStatusToggleConfirm?.cancelText
1004
- ? String(t(activeStatusToggleConfirm.cancelText))
1005
- : undefined
1006
- }
1007
- confirmText={
1008
- activeStatusToggleConfirm?.okText
1009
- ? String(t(activeStatusToggleConfirm.okText))
1010
- : undefined
1011
- }
1012
- danger={statusToggleConfirmDanger}
1013
- isOpen={statusToggleState.open}
1014
- message={
1015
- activeStatusToggleConfirm?.content
1016
- ? String(
1017
- t(
1018
- activeStatusToggleConfirm.content,
1019
- statusToggleContentInterpolation,
1020
- ),
1021
- )
1022
- : ''
1023
- }
1024
- title={String(t(activeStatusToggleConfirm?.title ?? 'common.tip'))}
1025
- onClose={closeStatusToggleConfirm}
1026
- onConfirm={handleConfirmStatusToggle}
1027
- />
1028
-
1029
- {SubmitModalComp && (
1030
- <SubmitModalComp
1031
- ref={submitModalRef}
1032
- onOk={async (values: unknown) => {
1033
- if (isDev) {
1034
- console.log('提交表单', values)
1035
- }
1036
- await fetchTableData?.()
1037
- }}
1038
- onCancel={() => {
1039
- // 取消时的清理逻辑(可选)
1040
- }}
1041
- />
1042
- )}
1043
-
1044
- {DetailModalComp && <DetailModalComp ref={detailModalRef} />}
1045
- </div>
1046
- )
1047
- }
1048
-
1049
- export default ListPageView