cx-chat 0.0.1 → 0.0.3

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 +97326 -0
  3. package/dist/cx-chat.umd.js +734 -0
  4. package/package.json +21 -1
  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 -264
  405. /package/{public → dist}/favicon.ico +0 -0
  406. /package/{public → dist}/vite.svg +0 -0
@@ -1,1094 +0,0 @@
1
- import { Form } from 'antd'
2
- import clsx from 'clsx'
3
- import React, {
4
- useCallback,
5
- useEffect,
6
- useId,
7
- useLayoutEffect,
8
- useMemo,
9
- useState,
10
- } from 'react'
11
- import { flushSync } from 'react-dom'
12
- import { useTranslation } from 'react-i18next'
13
-
14
- import { keysEqual } from '@/components/common/lite-table'
15
- import Pagination from '@/components/common/pagination'
16
- import { CxButton, CxCheckbox, CxEmptyState, CxRadio, CxTable, CxTooltip } from '@cx-ui'
17
- import { SearchArea } from './components/search-area'
18
- import {
19
- TABLE_VIEW_DEFAULT_PAGE_SIZE_OPTIONS,
20
- TABLE_VIEW_TREE_DEFAULTS,
21
- type ListRenderContext,
22
- type SearchButtonSize,
23
- type SearchField,
24
- type SelectTableViewProps,
25
- type TreeConfig,
26
- type TreeContext,
27
- } from './components/table-view.types'
28
- import { normalizeTreeAPIResult } from './components/tree-api-normalize'
29
- import { TreeSidebar } from './components/tree-sidebar'
30
-
31
- const isDev = import.meta.env.DEV
32
-
33
- function collectAllTreeNodeKeys(
34
- nodes: any[] | undefined,
35
- idKey: string,
36
- childrenKey: string,
37
- ): (string | number)[] {
38
- const keys: (string | number)[] = []
39
- const walk = (list: any[]) => {
40
- for (const n of list) {
41
- const id = n?.[idKey]
42
- if (id !== undefined && id !== null) keys.push(id)
43
- const children = n?.[childrenKey]
44
- if (Array.isArray(children) && children.length) walk(children)
45
- }
46
- }
47
- if (Array.isArray(nodes)) walk(nodes)
48
- return keys
49
- }
50
-
51
- function normalizeSearchValuesByFields(
52
- values: Record<string, any>,
53
- searchFields: SearchField[] | undefined,
54
- ) {
55
- const next: Record<string, any> = { ...values }
56
- const fields = searchFields ?? []
57
- for (const field of fields) {
58
- if (field.type !== 'dateRange') continue
59
- const startKey = field.rangeStartProp || `${field.prop}Start`
60
- const endKey = field.rangeEndProp || `${field.prop}End`
61
- const format = field.format || 'YYYY-MM-DD HH:mm:ss'
62
- const range = values[field.prop]
63
- if (Array.isArray(range) && range[0] && range[1]) {
64
- if (field.showTime) {
65
- next[startKey] = range[0].format(format)
66
- next[endKey] = range[1].format(format)
67
- } else {
68
- next[startKey] = range[0].startOf('day').format(format)
69
- next[endKey] = range[1].endOf('day').format(format)
70
- }
71
- } else {
72
- next[startKey] = undefined
73
- next[endKey] = undefined
74
- }
75
- }
76
- return next
77
- }
78
-
79
- /**
80
- * 无 ListPageView 顶栏,仅搜索 + 表格(+ 可选左侧树)。
81
- * 数据与请求与 `createSelectTableConfig` / `createListPageConfig` + `ListPageView` 同源,便于弹框内「选用户」等复用同一套 tableAPI / searchFields。
82
- * 单选:行点击始终选中该行(不二次点击取消);多选:行点击在已选/未选之间切换。
83
- */
84
- export default function SelectTableView({
85
- className,
86
- columns,
87
- defaultValue = null,
88
- expectTree = false,
89
- layout = 'vertical',
90
- leftWidth,
91
- noPadding = false,
92
- onChange,
93
- pageConfig,
94
- removeSelectedLabelKey = 'selectTable.remove_selected',
95
- rowKey = 'id',
96
- searchAreaExtra,
97
- searchSlot,
98
- selectedPanelTitleKey = 'selectTable.selected_panel',
99
- selectedPanelColumns,
100
- showSelectedPanel = true,
101
- treeCheckable = false,
102
- treeCheckedKeys,
103
- onTreeCheck,
104
- treeRenderLabel,
105
- selectionMode,
106
- value,
107
- }: SelectTableViewProps) {
108
- const { t } = useTranslation()
109
- const singleRadioName = `select-table-view-single-${useId()}`
110
- const [form] = Form.useForm()
111
-
112
- const isSingle = selectionMode === 'single'
113
- const isMultiple = selectionMode === 'multiple'
114
- const isNone = selectionMode === 'none'
115
- const isControlled = value !== undefined && typeof onChange === 'function'
116
-
117
- const [innerSingle, setInnerSingle] = useState<any | null>(() =>
118
- isSingle ? (defaultValue as any) ?? null : null,
119
- )
120
- const [innerMulti, setInnerMulti] = useState<any[]>(() =>
121
- !isSingle && Array.isArray(defaultValue) ? [...defaultValue] : [],
122
- )
123
-
124
- const selectedSingle = isSingle
125
- ? isControlled
126
- ? (value as any) ?? null
127
- : innerSingle
128
- : null
129
- const selectedMulti = !isSingle
130
- ? isControlled
131
- ? Array.isArray(value)
132
- ? value
133
- : []
134
- : innerMulti
135
- : []
136
-
137
- const setSelectedSingle = useCallback(
138
- (next: any | null) => {
139
- if (!isControlled) setInnerSingle(next)
140
- onChange?.(next)
141
- },
142
- [isControlled, onChange],
143
- )
144
-
145
- const setSelectedMulti = useCallback(
146
- (next: any[]) => {
147
- if (!isControlled) setInnerMulti(next)
148
- onChange?.(next)
149
- },
150
- [isControlled, onChange],
151
- )
152
-
153
- /** 多选:下方列表待批量移除的勾选 */
154
- const [bottomCheckedKeys, setBottomCheckedKeys] = useState<Set<string>>(
155
- () => new Set(),
156
- )
157
-
158
- const {
159
- fetchTableData,
160
- handleReset,
161
- handleSearch,
162
- handleTableChange,
163
- pageInfo = {},
164
- pageSizeOptions: pageSizeOptionsFromConfig,
165
- paginationData,
166
- patchSearchAndRefresh,
167
- replaceSearchAndRefresh,
168
- searchFields,
169
- searchFormData,
170
- setSearchFormData,
171
- table: tableConfig = {},
172
- tableData,
173
- tree,
174
- treeAPI,
175
- treeReloadSignal = 0,
176
- } = pageConfig
177
-
178
- const pageSizeOptions =
179
- pageSizeOptionsFromConfig?.length
180
- ? pageSizeOptionsFromConfig
181
- : [...TABLE_VIEW_DEFAULT_PAGE_SIZE_OPTIONS]
182
-
183
- const table = tableConfig || {}
184
- const tableScrollX =
185
- table.scrollX === false ? undefined : ((table.scrollX ?? 'max-content') as any)
186
- const searchButtonSize: SearchButtonSize =
187
- pageInfo.searchButtonSize ?? 'default'
188
-
189
- const hasTree = !!(treeAPI || tree)
190
- const mergedTree: TreeConfig | undefined = useMemo(() => {
191
- if (!hasTree) return undefined
192
- return {
193
- ...TABLE_VIEW_TREE_DEFAULTS,
194
- ...tree,
195
- title: tree?.title ?? pageInfo.title,
196
- }
197
- }, [hasTree, tree, pageInfo.title])
198
-
199
- useEffect(() => {
200
- if (expectTree && !treeAPI && isDev) {
201
- console.warn(
202
- '[SelectTableView] expectTree 为 true 但 pageConfig 未配置 treeAPI',
203
- )
204
- }
205
- }, [expectTree, treeAPI])
206
-
207
- const [treeData, setTreeData] = useState<any[]>([])
208
- const [treeDataLoaded, setTreeDataLoaded] = useState(!hasTree)
209
- const [expandedKeys, setExpandedKeys] = useState<Set<string | number>>(
210
- new Set(),
211
- )
212
- const [selectedKey, setSelectedKey] = useState<string | number | null>(null)
213
-
214
- const idKey = mergedTree?.fieldNames?.id ?? 'id'
215
- const childrenKey = mergedTree?.fieldNames?.children ?? 'children'
216
- const searchField = mergedTree?.searchField ?? 'id'
217
- const autoSearch = mergedTree?.autoSearch ?? true
218
- const unselectText = mergedTree?.unselectText ?? 'common.unselect_tree'
219
-
220
- useEffect(() => {
221
- if (!treeAPI) {
222
- setTreeDataLoaded(true)
223
- return
224
- }
225
- let mounted = true
226
- setTreeDataLoaded(false)
227
- treeAPI()
228
- .then((raw) => {
229
- if (!mounted) return
230
- setTreeData(normalizeTreeAPIResult(raw))
231
- setTreeDataLoaded(true)
232
- })
233
- .catch((err) => {
234
- if (!mounted) return
235
- console.error('[SelectTableView] Tree fetch error:', err)
236
- setTreeDataLoaded(true)
237
- })
238
- return () => {
239
- mounted = false
240
- }
241
- }, [treeAPI, hasTree, treeReloadSignal])
242
-
243
- useEffect(() => {
244
- if (!mergedTree?.defaultExpandAll || !treeData.length) return
245
- setExpandedKeys(
246
- new Set(collectAllTreeNodeKeys(treeData, idKey, childrenKey)),
247
- )
248
- }, [treeData, mergedTree?.defaultExpandAll, idKey, childrenKey])
249
-
250
- const findNodeByKey = useCallback(
251
- (nodes: any[], key: any): any | null => {
252
- for (const n of nodes) {
253
- if (n[idKey] === key) return n
254
- const children = n[childrenKey]
255
- if (Array.isArray(children)) {
256
- const found = findNodeByKey(children, key)
257
- if (found) return found
258
- }
259
- }
260
- return null
261
- },
262
- [idKey, childrenKey],
263
- )
264
-
265
- const toggleTreeExpand = useCallback((key: any, e?: React.MouseEvent) => {
266
- if (e) e.stopPropagation()
267
- setExpandedKeys((prev) => {
268
- const next = new Set(prev)
269
- if (next.has(key)) next.delete(key)
270
- else next.add(key)
271
- return next
272
- })
273
- }, [])
274
-
275
- const handleTreeSelect = useCallback(
276
- (key: any) => {
277
- const formSnap = normalizeSearchValuesByFields(
278
- form.getFieldsValue(true),
279
- searchFields ?? [],
280
- )
281
- const mergedBase = { ...(searchFormData || {}), ...formSnap }
282
-
283
- if (key == null) {
284
- setSelectedKey(null)
285
- mergedTree?.onSelect?.(null)
286
- if (!mergedTree?.searchField) return
287
- const next = { ...mergedBase }
288
- delete next[searchField]
289
- if (replaceSearchAndRefresh && autoSearch) {
290
- replaceSearchAndRefresh(next)
291
- } else if (setSearchFormData) {
292
- flushSync(() => {
293
- setSearchFormData(next)
294
- })
295
- if (autoSearch) handleSearch()
296
- }
297
- return
298
- }
299
-
300
- setSelectedKey(key)
301
- const node = findNodeByKey(treeData, key)
302
- if (!node) return
303
- mergedTree?.onSelect?.(node)
304
- if (!mergedTree?.searchField) return
305
- const full = { ...mergedBase, [searchField]: node[idKey] }
306
- if (patchSearchAndRefresh && autoSearch) {
307
- patchSearchAndRefresh(full)
308
- } else if (setSearchFormData) {
309
- flushSync(() => {
310
- setSearchFormData(full)
311
- })
312
- if (autoSearch) handleSearch()
313
- }
314
- },
315
- [
316
- treeData,
317
- findNodeByKey,
318
- form,
319
- mergedTree,
320
- searchField,
321
- idKey,
322
- autoSearch,
323
- patchSearchAndRefresh,
324
- replaceSearchAndRefresh,
325
- searchFields,
326
- searchFormData,
327
- setSearchFormData,
328
- handleSearch,
329
- ],
330
- )
331
-
332
- useEffect(() => {
333
- if (mergedTree?.defaultSelectRoot === false) return
334
- if (!treeDataLoaded || treeData.length === 0) return
335
- if (selectedKey != null) return
336
- const root = treeData[0]
337
- const key = root?.[idKey]
338
- if (key === undefined || key === null) return
339
- handleTreeSelect(key)
340
- // eslint-disable-next-line react-hooks/exhaustive-deps -- 树就绪且无选中时一次
341
- }, [
342
- mergedTree?.defaultSelectRoot,
343
- treeDataLoaded,
344
- treeData,
345
- idKey,
346
- selectedKey,
347
- handleTreeSelect,
348
- ])
349
-
350
- const selectedNode =
351
- selectedKey != null ? findNodeByKey(treeData, selectedKey) : null
352
- const treeCtx: TreeContext | undefined = mergedTree
353
- ? { selectedKey, selectedNode, treeData, treeDataLoaded }
354
- : undefined
355
-
356
- React.useEffect(() => {
357
- if (pageConfig) {
358
- ;(pageConfig as any).treeContext = treeCtx
359
- }
360
- }, [treeCtx, pageConfig])
361
-
362
- useLayoutEffect(() => {
363
- form.setFieldsValue(
364
- normalizeSearchValuesByFields(searchFormData || {}, searchFields),
365
- )
366
- }, [form, searchFields, searchFormData])
367
-
368
- const applySearchFromForm = () => {
369
- const vals = form.getFieldsValue(true)
370
- let normalizedVals = normalizeSearchValuesByFields(vals, searchFields)
371
- const treeListSearchKey =
372
- mergedTree?.searchField ??
373
- (treeAPI ? TABLE_VIEW_TREE_DEFAULTS.searchField : undefined)
374
- if (
375
- treeListSearchKey &&
376
- searchFormData &&
377
- Object.prototype.hasOwnProperty.call(searchFormData, treeListSearchKey)
378
- ) {
379
- normalizedVals = {
380
- ...normalizedVals,
381
- [treeListSearchKey]: searchFormData[treeListSearchKey],
382
- }
383
- }
384
- flushSync(() => {
385
- setSearchFormData?.(normalizedVals)
386
- })
387
- handleSearch()
388
- }
389
-
390
- const onReset = () => {
391
- handleReset()
392
- }
393
-
394
- const renderCell = (col: any, record: any, rowIndex: number) => {
395
- const fieldKey = col.dataIndex ?? col.key
396
- if (col.render) {
397
- return col.render(record[fieldKey], record, rowIndex)
398
- }
399
- const raw = record[fieldKey]
400
- if (col.ellipsis) {
401
- const s = raw != null && raw !== '' ? String(raw) : '-'
402
- const inner = (
403
- <div className="min-w-0 max-w-full overflow-hidden">
404
- <span className="block truncate">{s}</span>
405
- </div>
406
- )
407
- if (s === '-') return inner
408
- const placement = col.ellipsisTooltipPlacement ?? 'right'
409
- const tw = col.ellipsisTooltipWidth
410
- const maxW =
411
- tw == null || tw === ''
412
- ? '300px'
413
- : typeof tw === 'number'
414
- ? `${tw}px`
415
- : String(tw)
416
- return (
417
- <CxTooltip
418
- placement={placement}
419
- styles={{
420
- root: { maxWidth: maxW },
421
- }}
422
- title={
423
- <div
424
- className="max-h-[50vh] overflow-y-auto whitespace-pre-wrap break-words text-left"
425
- style={{ maxWidth: maxW }}
426
- >
427
- {s}
428
- </div>
429
- }
430
- >
431
- {inner}
432
- </CxTooltip>
433
- )
434
- }
435
- return raw
436
- }
437
-
438
- const listCtxBase: ListRenderContext = {
439
- columns: columns ?? [],
440
- renderCell,
441
- tableData,
442
- t,
443
- tree: treeCtx,
444
- }
445
-
446
- const showListArea =
447
- !mergedTree ||
448
- !!selectedKey ||
449
- mergedTree?.sameNodeReselect === 'deselect'
450
- const showNoDataEmpty = showListArea && paginationData.total === 0
451
-
452
- const searchAreaExtraNode =
453
- searchAreaExtra == null || searchAreaExtra === false
454
- ? null
455
- : typeof searchAreaExtra === 'function'
456
- ? (searchAreaExtra as (ctx: any) => React.ReactNode)({
457
- tree: treeCtx,
458
- searchButtonSize,
459
- })
460
- : searchAreaExtra
461
-
462
- const singleColumns = useMemo(() => {
463
- if (!isSingle) return columns
464
- const selectCol = {
465
- key: '__select',
466
- title: '',
467
- width: 48,
468
- render: (_: unknown, data: any) => {
469
- const k = data?.[rowKey]
470
- const sel =
471
- selectedSingle &&
472
- keysEqual(selectedSingle?.[rowKey], k)
473
- return (
474
- <CxRadio
475
- name={singleRadioName}
476
- checked={!!sel}
477
- onClick={(e) => e.stopPropagation()}
478
- onChange={() => setSelectedSingle(data)}
479
- />
480
- )
481
- },
482
- }
483
- return [selectCol, ...columns] as any[]
484
- }, [isSingle, columns, rowKey, selectedSingle, setSelectedSingle, t])
485
-
486
- // selectionMode 为 'none' 时,不添加选择列
487
- const displayColumns = isNone ? columns : singleColumns
488
-
489
- const toggleMultiTop = useCallback(
490
- (record: any) => {
491
- const k = record?.[rowKey]
492
- const exists = selectedMulti.some((r) =>
493
- keysEqual(r?.[rowKey], k),
494
- )
495
- if (exists) {
496
- setSelectedMulti(selectedMulti.filter((r) => !keysEqual(r?.[rowKey], k)))
497
- } else {
498
- setSelectedMulti([...selectedMulti, record])
499
- }
500
- },
501
- [rowKey, selectedMulti, setSelectedMulti],
502
- )
503
-
504
- /** 多选:全选/取消全选当前页(仅影响上方列表当前 tableData) */
505
- const toggleMultiSelectAllCurrentPage = useCallback(
506
- (checked: boolean) => {
507
- if (!Array.isArray(tableData) || tableData.length === 0) return
508
- if (!checked) {
509
- setSelectedMulti(
510
- selectedMulti.filter(
511
- (r) =>
512
- !tableData.some((row) => keysEqual(row?.[rowKey], r?.[rowKey])),
513
- ),
514
- )
515
- return
516
- }
517
-
518
- const next = [...selectedMulti]
519
- for (const row of tableData) {
520
- const k = row?.[rowKey]
521
- const exists = next.some((r) => keysEqual(r?.[rowKey], k))
522
- if (!exists) next.push(row)
523
- }
524
- setSelectedMulti(next)
525
- },
526
- [rowKey, selectedMulti, setSelectedMulti, tableData],
527
- )
528
-
529
- /** 单选:每次点击行都是选中该行(同一条再点仍为选中,不取消) */
530
- const selectSingleRow = useCallback(
531
- (record: any) => {
532
- setSelectedSingle(record)
533
- },
534
- [setSelectedSingle],
535
- )
536
-
537
- const bottomToggleCheck = useCallback((record: any) => {
538
- const k = String(record?.[rowKey])
539
- setBottomCheckedKeys((prev) => {
540
- const next = new Set(prev)
541
- if (next.has(k)) next.delete(k)
542
- else next.add(k)
543
- return next
544
- })
545
- }, [rowKey])
546
-
547
- /** 已选择区:全选/取消全选(用于批量移除的勾选) */
548
- const toggleBottomSelectAll = useCallback((checked: boolean) => {
549
- if (!checked) {
550
- setBottomCheckedKeys(new Set())
551
- return
552
- }
553
- setBottomCheckedKeys(
554
- new Set(selectedMulti.map((r) => String(r?.[rowKey]))),
555
- )
556
- }, [rowKey, selectedMulti])
557
-
558
- const removeBottomChecked = useCallback(() => {
559
- if (bottomCheckedKeys.size === 0) return
560
- const next = selectedMulti.filter(
561
- (r) => !bottomCheckedKeys.has(String(r?.[rowKey])),
562
- )
563
- setSelectedMulti(next)
564
- setBottomCheckedKeys(new Set())
565
- }, [bottomCheckedKeys, rowKey, selectedMulti, setSelectedMulti])
566
-
567
- const multiBottomColumns = useMemo(() => {
568
- const bottomBase = (selectedPanelColumns?.length ? selectedPanelColumns : columns) as any[]
569
- const bottomAllChecked =
570
- selectedMulti.length > 0 &&
571
- selectedMulti.every((r) => bottomCheckedKeys.has(String(r?.[rowKey])))
572
- const checkCol = {
573
- key: '__chk',
574
- title: (
575
- <div className="flex items-center justify-center">
576
- <CxCheckbox
577
- checked={bottomAllChecked}
578
- size="md"
579
- aria-label={t('selectTable.select_all')}
580
- onChange={(checked) => toggleBottomSelectAll(checked)}
581
- />
582
- </div>
583
- ),
584
- width: 44,
585
- onHeaderCell: () => ({ className: 'cx-table__select-col' }),
586
- onCell: () => ({ className: 'cx-table__select-col' }),
587
- render: (_: unknown, data: any) => {
588
- const k = String(data?.[rowKey])
589
- return (
590
- <div className="flex items-center justify-center">
591
- <CxCheckbox
592
- checked={bottomCheckedKeys.has(k)}
593
- size="md"
594
- aria-label={t('selectTable.mark_remove')}
595
- onChange={() => bottomToggleCheck(data)}
596
- />
597
- </div>
598
- )
599
- },
600
- }
601
- return [checkCol, ...(bottomBase ?? [])] as any[]
602
- }, [
603
- selectedPanelColumns,
604
- columns,
605
- rowKey,
606
- selectedMulti,
607
- bottomCheckedKeys,
608
- bottomToggleCheck,
609
- toggleBottomSelectAll,
610
- t,
611
- ])
612
-
613
- /** 多选:上方列表首列复选框,与行点击一致;点选框时阻止冒泡避免与行点击重复触发 */
614
- const multiTopColumns = useMemo(() => {
615
- const pageAllSelected =
616
- Array.isArray(tableData) &&
617
- tableData.length > 0 &&
618
- tableData.every((row) =>
619
- selectedMulti.some((r) => keysEqual(r?.[rowKey], row?.[rowKey])),
620
- )
621
- const checkCol = {
622
- key: '__select',
623
- title: (
624
- <div className="flex items-center justify-center">
625
- <CxCheckbox
626
- checked={pageAllSelected}
627
- size="md"
628
- aria-label={t('selectTable.select_all')}
629
- onChange={(checked) => toggleMultiSelectAllCurrentPage(checked)}
630
- />
631
- </div>
632
- ),
633
- width: 48,
634
- onHeaderCell: () => ({ className: 'cx-table__select-col' }),
635
- onCell: () => ({ className: 'cx-table__select-col' }),
636
- render: (_: unknown, data: any) => {
637
- const k = data?.[rowKey]
638
- const sel = selectedMulti.some((r) =>
639
- keysEqual(r?.[rowKey], k),
640
- )
641
- return (
642
- <div className="flex items-center justify-center">
643
- <CxCheckbox
644
- checked={sel}
645
- size="md"
646
- aria-label={t('selectTable.select_row')}
647
- onChange={() => toggleMultiTop(data)}
648
- />
649
- </div>
650
- )
651
- },
652
- }
653
- return [checkCol, ...columns] as any[]
654
- }, [
655
- columns,
656
- rowKey,
657
- selectedMulti,
658
- t,
659
- tableData,
660
- toggleMultiTop,
661
- toggleMultiSelectAllCurrentPage,
662
- ])
663
-
664
- const multiListCtx: ListRenderContext = {
665
- ...listCtxBase,
666
- columns: multiBottomColumns,
667
- tableData: selectedMulti,
668
- }
669
-
670
- /** 多选:上方列表上下文,带复选框 */
671
- const multiTopListCtx: ListRenderContext = {
672
- ...listCtxBase,
673
- columns: multiTopColumns,
674
- }
675
-
676
- const singleListCtx: ListRenderContext = {
677
- ...listCtxBase,
678
- columns: displayColumns,
679
- }
680
-
681
- return (
682
- <div
683
- className={clsx(
684
- 'select-table flex min-h-0 h-full w-full min-w-0 flex-col',
685
- mergedTree ? 'select-table-tree' : '',
686
- isMultiple ? 'select-table-multiple' : '',
687
- className,
688
- )}>
689
- <div
690
- className={clsx(
691
- 'flex min-h-0 w-full flex-1 flex-col',
692
- noPadding ? '' : 'p-5 pb-px',
693
- mergedTree && 'flex-row gap-4 items-stretch',
694
- )}>
695
- {mergedTree ? (
696
- <div className="select-table-sidebar">
697
- <TreeSidebar
698
- expandedKeys={expandedKeys}
699
- onSelect={handleTreeSelect}
700
- onToggleExpand={toggleTreeExpand}
701
- selectedKey={selectedKey}
702
- tree={mergedTree}
703
- treeData={treeData}
704
- checkable={treeCheckable}
705
- checkedKeys={treeCheckedKeys}
706
- onCheck={onTreeCheck}
707
- renderLabel={treeRenderLabel}
708
- />
709
- </div>
710
- ) : null}
711
- <div
712
- className={clsx(
713
- mergedTree ? 'flex-1 min-w-0 h-full' : 'w-full',
714
- 'flex flex-col min-h-0',
715
- // 外层纵向滚动会带动整张表,antd 表头固定依赖表体独立滚动
716
- mergedTree ? 'overflow-hidden' : '',
717
- isSingle || layout === 'horizontal' ? 'h-full' : '',
718
- 'select-table-view-right',
719
- )}>
720
- {isSingle ? (
721
- <div className="flex min-h-[320px] flex-1 flex-col overflow-hidden rounded-xl border border-gray-200 bg-white shadow-sm dark:border-gray-700 dark:bg-gray-800">
722
- {searchSlot ? (
723
- searchSlot
724
- ) : ((mergedTree &&
725
- (selectedKey != null ||
726
- mergedTree?.sameNodeReselect === 'deselect')) ||
727
- !mergedTree) &&
728
- searchFields &&
729
- searchFields.length > 0 ? (
730
- <SearchArea
731
- extra={searchAreaExtraNode}
732
- forceHorizontal
733
- form={form}
734
- listTitle={pageInfo.listTitle}
735
- onReset={onReset}
736
- onSearch={applySearchFromForm}
737
- searchButtonSize={searchButtonSize}
738
- searchFields={searchFields}
739
- searchFormData={searchFormData}
740
- total={paginationData.total}
741
- />
742
- ) : null}
743
- <>
744
- <div className="relative min-h-0 flex-1 overflow-hidden">
745
- {pageConfig.loading && (
746
- <div className="absolute inset-0 z-[1] flex items-center justify-center bg-white/50 text-sm text-gray-500 dark:bg-gray-900/40">
747
- {t('common.loading')}
748
- </div>
749
- )}
750
- {showListArea && (!showNoDataEmpty || tableData.length) ? (
751
- <div
752
- className={clsx(
753
- 'min-h-0 flex-1 min-w-0 h-full',
754
- tableScrollX ? 'overflow-x-auto' : 'overflow-x-hidden',
755
- )}>
756
- <CxTable
757
- size="small"
758
- className="h-full"
759
- autoScrollY
760
- minScrollY={160}
761
- sticky
762
- dataSource={tableData}
763
- rowKey={rowKey}
764
- // 透传列配置,避免丢失 minWidth/ellipsis/fixed/onCell 等属性
765
- columns={(singleListCtx.columns ?? []).map((col: any) => ({
766
- ...col,
767
- key: col.key ?? col.dataIndex,
768
- dataIndex: col.dataIndex ?? col.key,
769
- title:
770
- typeof col.title === 'string'
771
- ? (col.title ? t(col.title) : '')
772
- : col.title,
773
- render: (_: unknown, record: any, rowIndex: number) =>
774
- renderCell(col, record, rowIndex),
775
- }))}
776
- onRow={(record: any) => ({
777
- onClick: isNone ? undefined : () => selectSingleRow(record),
778
- })}
779
- rowClassName={(record: any) => {
780
- if (isNone) return ''
781
- return selectedSingle &&
782
- keysEqual(
783
- record?.[rowKey],
784
- selectedSingle?.[rowKey],
785
- )
786
- ? 'bg-blue-50 dark:bg-blue-900/20'
787
- : ''
788
- }}
789
- tableLayout={tableScrollX ? 'auto' : 'fixed'}
790
- scroll={tableScrollX ? ({ x: tableScrollX } as any) : undefined}
791
- />
792
- </div>
793
- ) : null}
794
- {mergedTree &&
795
- selectedKey == null &&
796
- mergedTree?.sameNodeReselect !== 'deselect' ? (
797
- <CxEmptyState message={t(unselectText)} />
798
- ) : null}
799
- {showNoDataEmpty && !tableData.length && (
800
- <CxEmptyState message={t('common.no_data')} />
801
- )}
802
- </div>
803
- {table.showPagination !== false && paginationData.total > 0 && (
804
- <div className="shrink-0 border-t border-gray-200 px-3 py-3 dark:border-gray-700">
805
- <Pagination
806
- onPageChange={(page) => {
807
- handleTableChange({
808
- current: page,
809
- pageSize: paginationData.pageSize,
810
- })
811
- }}
812
- onPageSizeChange={(size) => {
813
- handleTableChange({
814
- current: paginationData.page,
815
- pageSize: size,
816
- })
817
- }}
818
- page={paginationData.page}
819
- pageSize={paginationData.pageSize}
820
- pageSizeOptions={pageSizeOptions}
821
- showSizeChanger
822
- total={paginationData.total}
823
- zeroIndexed={false}
824
- />
825
- </div>
826
- )}
827
- </>
828
- </div>
829
- ) : (
830
- <div className={clsx(
831
- "flex min-h-0 flex-1 gap-4",
832
- layout === 'horizontal' ? 'flex-row h-full' : 'flex-col'
833
- )}>
834
- {/* 左侧:搜索区 + 列表 */}
835
- <div
836
- className="flex flex-col min-w-0 h-full"
837
- style={{
838
- width: layout === 'horizontal' ? (typeof leftWidth === 'number' ? `${leftWidth}px` : leftWidth || '50%') : undefined,
839
- flex: layout === 'horizontal' ? 'none' : 1,
840
- }}
841
- >
842
- <div className="flex h-full shrink-0 flex-col overflow-hidden rounded-xl border border-gray-200 bg-white shadow-sm dark:border-gray-700 dark:bg-gray-800">
843
- {searchSlot ? (
844
- <div className="shrink-0 border-b border-gray-200 dark:border-gray-700">
845
- {searchSlot}
846
- </div>
847
- ) : ((mergedTree &&
848
- (selectedKey != null ||
849
- mergedTree?.sameNodeReselect === 'deselect')) ||
850
- !mergedTree) &&
851
- searchFields &&
852
- searchFields.length > 0 ? (
853
- <SearchArea
854
- extra={searchAreaExtraNode}
855
- forceHorizontal
856
- form={form}
857
- listTitle={pageInfo.listTitle}
858
- onReset={onReset}
859
- onSearch={applySearchFromForm}
860
- searchButtonSize={searchButtonSize}
861
- searchFields={searchFields}
862
- searchFormData={searchFormData}
863
- total={paginationData.total}
864
- />
865
- ) : null}
866
- <div className="relative min-h-0 flex-1 overflow-hidden">
867
- {pageConfig.loading && (
868
- <div className="absolute inset-0 z-[1] flex items-center justify-center bg-white/50 text-sm dark:bg-gray-900/40">
869
- {t('common.loading')}
870
- </div>
871
- )}
872
- {showListArea && (!showNoDataEmpty || tableData.length) ? (
873
- <div
874
- className={clsx(
875
- 'min-h-0 flex-1 min-w-0 h-full',
876
- tableScrollX ? 'overflow-x-auto' : 'overflow-x-hidden',
877
- )}>
878
- <CxTable
879
- size="small"
880
- className="h-full"
881
- autoScrollY
882
- minScrollY={isNone ? 160 : 400}
883
- sticky
884
- dataSource={tableData}
885
- rowKey={rowKey}
886
- // 透传列配置,避免丢失 minWidth/ellipsis/fixed/onCell 等属性
887
- columns={((isMultiple ? multiTopListCtx.columns : singleListCtx.columns) ?? []).map((col: any) => ({
888
- ...col,
889
- key: col.key ?? col.dataIndex,
890
- dataIndex: col.dataIndex ?? col.key,
891
- title:
892
- typeof col.title === 'string'
893
- ? (col.title ? t(col.title) : '')
894
- : col.title,
895
- render: (_: unknown, record: any, rowIndex: number) =>
896
- renderCell(col, record, rowIndex),
897
- }))}
898
- onRow={
899
- isMultiple
900
- ? (record: any) => ({
901
- onClick: () => toggleMultiTop(record),
902
- })
903
- : isSingle
904
- ? (record: any) => ({
905
- onClick: () => selectSingleRow(record),
906
- })
907
- : undefined
908
- }
909
- rowClassName={
910
- isMultiple
911
- ? (record: any) =>
912
- selectedMulti.some((r) =>
913
- keysEqual(r?.[rowKey], record?.[rowKey]),
914
- )
915
- ? 'bg-blue-50 dark:bg-blue-900/20'
916
- : ''
917
- : undefined
918
- }
919
- tableLayout={tableScrollX ? 'auto' : 'fixed'}
920
- scroll={tableScrollX ? ({ x: tableScrollX } as any) : undefined}
921
- />
922
- </div>
923
- ) : null}
924
- {mergedTree &&
925
- selectedKey == null &&
926
- mergedTree?.sameNodeReselect !== 'deselect' ? (
927
- <CxEmptyState message={t(unselectText)} />
928
- ) : null}
929
- {showNoDataEmpty && !tableData.length && (
930
- <CxEmptyState message={t('common.no_data')} />
931
- )}
932
- </div>
933
- {table.showPagination !== false && paginationData.total > 0 && (
934
- <div className="shrink-0 border-t border-gray-200 px-3 py-4 dark:border-gray-700">
935
- <Pagination
936
- onPageChange={(page) => {
937
- handleTableChange({
938
- current: page,
939
- pageSize: paginationData.pageSize,
940
- })
941
- }}
942
- onPageSizeChange={(size) => {
943
- handleTableChange({
944
- current: paginationData.page,
945
- pageSize: size,
946
- })
947
- }}
948
- page={paginationData.page}
949
- pageSize={paginationData.pageSize}
950
- pageSizeOptions={pageSizeOptions}
951
- showSizeChanger
952
- total={paginationData.total}
953
- zeroIndexed={false}
954
- />
955
- </div>
956
- )}
957
- </div>
958
- </div>
959
-
960
- {/* 右侧:已选列表(仅横向布局且非none模式) */}
961
- {showSelectedPanel && layout === 'horizontal' && !isNone && (
962
- <div className="flex-1 min-w-0 flex flex-col h-full">
963
- <div className="flex flex-1 flex-col overflow-hidden rounded-xl border border-gray-200 bg-white shadow-sm dark:border-gray-700 dark:bg-gray-800">
964
- <div className="shrink-0 border-b border-gray-100 px-3 py-2 dark:border-gray-700">
965
- <div className="flex items-center justify-between gap-2">
966
- <h4 className="min-w-0 text-sm font-medium text-gray-600 dark:text-gray-300">
967
- {t(selectedPanelTitleKey)}{' '}
968
- <span className="text-gray-400">({selectedMulti.length})</span>
969
- </h4>
970
- <CxButton
971
- type="default"
972
- size="small"
973
- disabled={bottomCheckedKeys.size === 0}
974
- onClick={removeBottomChecked}>
975
- {t(removeSelectedLabelKey)}
976
- </CxButton>
977
- </div>
978
- </div>
979
- <div
980
- className={clsx(
981
- 'min-h-0 flex-1 overflow-y-auto',
982
- tableScrollX ? 'overflow-x-auto' : 'overflow-x-hidden',
983
- )}>
984
- {selectedMulti.length === 0 ? (
985
- <div className="px-3 py-8 text-center text-sm text-gray-400">
986
- {t('selectTable.empty_selected')}
987
- </div>
988
- ) : (
989
- <CxTable
990
- size="small"
991
- className="h-full"
992
- autoScrollY
993
- minScrollY={80}
994
- sticky
995
- dataSource={selectedMulti}
996
- rowKey={rowKey}
997
- // 透传列配置,避免丢失 minWidth/ellipsis/fixed/onCell 等属性
998
- columns={(multiListCtx.columns ?? []).map((col: any) => ({
999
- ...col,
1000
- key: col.key ?? col.dataIndex,
1001
- dataIndex: col.dataIndex ?? col.key,
1002
- title:
1003
- typeof col.title === 'string'
1004
- ? (col.title ? t(col.title) : '')
1005
- : col.title,
1006
- render: (_: unknown, record: any, rowIndex: number) =>
1007
- renderCell(col, record, rowIndex),
1008
- }))}
1009
- onRow={(record: any) => ({
1010
- onClick: () => bottomToggleCheck(record),
1011
- })}
1012
- rowClassName={(record: any) =>
1013
- bottomCheckedKeys.has(String(record?.[rowKey]))
1014
- ? 'bg-amber-50 dark:bg-amber-900/20'
1015
- : ''
1016
- }
1017
- tableLayout={tableScrollX ? 'auto' : 'fixed'}
1018
- scroll={tableScrollX ? ({ x: tableScrollX } as any) : undefined}
1019
- />
1020
- )}
1021
- </div>
1022
- </div>
1023
- </div>
1024
- )}
1025
- </div>
1026
- )}
1027
- {/* 纵向布局:已选面板放在右侧列内,避免与左侧树同属最外层横向 flex 时变成第三列 */}
1028
- {showSelectedPanel && !isSingle && !isNone && layout === 'vertical' && (
1029
- <div className="mt-3 flex h-[520px] min-h-[520px] shrink-0 flex-col overflow-hidden rounded-xl border border-gray-200 bg-white shadow-sm dark:border-gray-700 dark:bg-gray-800">
1030
- <div className="shrink-0 border-b border-gray-100 px-3 py-2 dark:border-gray-700">
1031
- <div className="flex items-center justify-between gap-2">
1032
- <h4 className="min-w-0 text-sm font-medium text-gray-600 dark:text-gray-300">
1033
- {t(selectedPanelTitleKey)}{' '}
1034
- <span className="text-gray-400">({selectedMulti.length})</span>
1035
- </h4>
1036
- <CxButton
1037
- type="default"
1038
- size="small"
1039
- disabled={bottomCheckedKeys.size === 0}
1040
- onClick={removeBottomChecked}>
1041
- {t(removeSelectedLabelKey)}
1042
- </CxButton>
1043
- </div>
1044
- </div>
1045
- <div
1046
- className={clsx(
1047
- 'min-h-0 flex-1 overflow-y-auto',
1048
- tableScrollX ? 'overflow-x-auto' : 'overflow-x-hidden',
1049
- )}>
1050
- {selectedMulti.length === 0 ? (
1051
- <div className="px-3 py-8 text-center text-sm text-gray-400">
1052
- {t('selectTable.empty_selected')}
1053
- </div>
1054
- ) : (
1055
- <CxTable
1056
- size="small"
1057
- className="h-full"
1058
- autoScrollY
1059
- minScrollY={80}
1060
- sticky
1061
- dataSource={selectedMulti}
1062
- rowKey={rowKey}
1063
- // 透传列配置,避免丢失 minWidth/ellipsis/fixed/onCell 等属性
1064
- columns={(multiListCtx.columns ?? []).map((col: any) => ({
1065
- ...col,
1066
- key: col.key ?? col.dataIndex,
1067
- dataIndex: col.dataIndex ?? col.key,
1068
- title:
1069
- typeof col.title === 'string'
1070
- ? (col.title ? t(col.title) : '')
1071
- : col.title,
1072
- render: (_: unknown, record: any, rowIndex: number) =>
1073
- renderCell(col, record, rowIndex),
1074
- }))}
1075
- onRow={(record: any) => ({
1076
- onClick: () => bottomToggleCheck(record),
1077
- })}
1078
- rowClassName={(record: any) =>
1079
- bottomCheckedKeys.has(String(record?.[rowKey]))
1080
- ? 'bg-amber-50 dark:bg-amber-900/20'
1081
- : ''
1082
- }
1083
- tableLayout={tableScrollX ? 'auto' : 'fixed'}
1084
- scroll={tableScrollX ? ({ x: tableScrollX } as any) : undefined}
1085
- />
1086
- )}
1087
- </div>
1088
- </div>
1089
- )}
1090
- </div>
1091
- </div>
1092
- </div>
1093
- )
1094
- }