@gadmin2n/schematics 0.0.68 → 0.0.70

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 (223) hide show
  1. package/dist/lib/application/files/gadmin2-game-angle-demo/config/prisma/canvas.prisma +45 -0
  2. package/dist/lib/application/files/gadmin2-game-angle-demo/config/prisma/example.prisma +24 -23
  3. package/dist/lib/application/files/gadmin2-game-angle-demo/config/prisma/system.prisma +93 -90
  4. package/dist/lib/application/files/gadmin2-game-angle-demo/config/ui/Audit.ts +70 -0
  5. package/dist/lib/application/files/gadmin2-game-angle-demo/config/ui/Event.ts +3 -3
  6. package/dist/lib/application/files/gadmin2-game-angle-demo/config/ui/Page.ts +70 -0
  7. package/dist/lib/application/files/gadmin2-game-angle-demo/config/ui/PageResource.ts +70 -0
  8. package/dist/lib/application/files/gadmin2-game-angle-demo/config/ui/Resource.ts +70 -0
  9. package/dist/lib/application/files/gadmin2-game-angle-demo/config/ui/RolePages.ts +70 -0
  10. package/dist/lib/application/files/gadmin2-game-angle-demo/config/ui/RoleResource.ts +70 -0
  11. package/dist/lib/application/files/gadmin2-game-angle-demo/config/ui/SavedQuery.ts +70 -0
  12. package/dist/lib/application/files/gadmin2-game-angle-demo/config/ui/User.ts +70 -0
  13. package/dist/lib/application/files/gadmin2-game-angle-demo/readme.md +95 -2
  14. package/dist/lib/application/files/gadmin2-game-angle-demo/server/.eslintrc.js +4 -2
  15. package/dist/lib/application/files/gadmin2-game-angle-demo/server/.prettierignore +7 -1
  16. package/dist/lib/application/files/gadmin2-game-angle-demo/server/config/index.ts +1 -3
  17. package/dist/lib/application/files/gadmin2-game-angle-demo/server/gadmin-cli.json +2 -2
  18. package/dist/lib/application/files/gadmin2-game-angle-demo/server/package.json +6 -5
  19. package/dist/lib/application/files/gadmin2-game-angle-demo/server/scripts/lib/page-helpers.ts +157 -0
  20. package/dist/lib/application/files/gadmin2-game-angle-demo/server/scripts/page-manage.ts +290 -0
  21. package/dist/lib/application/files/gadmin2-game-angle-demo/server/scripts/permission-manage.ts +184 -0
  22. package/dist/lib/application/files/gadmin2-game-angle-demo/server/scripts/prismaModels.ts +143 -0
  23. package/dist/lib/application/files/gadmin2-game-angle-demo/server/scripts/sync-resources.ts +100 -0
  24. package/dist/lib/application/files/gadmin2-game-angle-demo/server/scripts/syncDataMngtPages.ts +119 -0
  25. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/index.ts +1 -3
  26. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/permissions.ts +109 -143
  27. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/main.ts +5 -2
  28. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/canvas/canvas.controller.spec.ts +20 -0
  29. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/canvas/canvas.controller.ts +81 -0
  30. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/canvas/canvas.dto.ts +49 -0
  31. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/canvas/canvas.module.ts +11 -0
  32. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/canvas/canvas.service.spec.ts +334 -0
  33. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/canvas/canvas.service.ts +249 -0
  34. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/index.ts +2 -1
  35. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/page/page.service.ts +2 -2
  36. package/dist/lib/application/files/gadmin2-game-angle-demo/server/src/modules/role/role.service.ts +17 -4
  37. package/dist/lib/application/files/gadmin2-game-angle-demo/server/yarn.lock +7737 -0
  38. package/dist/lib/application/files/gadmin2-game-angle-demo/web/.env +1 -0
  39. package/dist/lib/application/files/gadmin2-game-angle-demo/web/.eslintrc.json +38 -0
  40. package/dist/lib/application/files/gadmin2-game-angle-demo/web/.prettierignore +2 -0
  41. package/dist/lib/application/files/gadmin2-game-angle-demo/web/dev-shell-entry.html +49 -0
  42. package/dist/lib/application/files/gadmin2-game-angle-demo/web/package.json +13 -2
  43. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/App.tsx +137 -111
  44. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/auditLogProvider.ts +46 -44
  45. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/authProvider.ts +32 -12
  46. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/BulkActions.tsx +74 -23
  47. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/ListPageHeader.tsx +9 -9
  48. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/RowActions.tsx +57 -31
  49. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/SearchBar.tsx +53 -24
  50. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/SqlModal.tsx +148 -72
  51. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/AgentContext.tsx +190 -16
  52. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/ElementInspector.tsx +914 -63
  53. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/agentAttributes.ts +17 -8
  54. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/index.ts +0 -10
  55. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/inspectorActions.ts +969 -46
  56. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/pagePathUtils.ts +66 -0
  57. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/promptGenerator.ts +85 -25
  58. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/propPatcher.ts +84 -0
  59. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/auditLog/components/eventList/index.tsx +16 -20
  60. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/auditLog/components/index.ts +4 -4
  61. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/auditLog/components/logButton/index.tsx +13 -13
  62. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/auditLog/components/logList/index.tsx +15 -16
  63. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/auditLog/components/modalDiffViewer/index.tsx +15 -15
  64. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/auditLog/index.tsx +2 -2
  65. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/BarChart/index.tsx +896 -0
  66. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/ChartSwitcher/index.tsx +219 -0
  67. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/ChartViewer/index.tsx +159 -0
  68. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/Filter/index.tsx +192 -0
  69. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/LineChart/index.tsx +1034 -0
  70. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/NumCard/NumCard.module.css +8 -0
  71. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/NumCard/index.tsx +509 -0
  72. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/NumLineCard/index.tsx +66 -0
  73. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/PieChart/index.tsx +552 -0
  74. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/RadarChart/index.tsx +263 -0
  75. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/Section/index.tsx +35 -0
  76. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/Table/index.tsx +207 -0
  77. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/TreemapChart/index.tsx +382 -0
  78. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/WorldMap/index.tsx +135 -0
  79. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/chart-constants.ts +53 -0
  80. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/icon/InfoIcon.tsx +8 -0
  81. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/icon/index.ts +1 -0
  82. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/map/config.ts +31 -0
  83. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/map/nameMap.json +9 -0
  84. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/map/world.geo.json +39349 -0
  85. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/canvas/metric-info-tooltip/index.tsx +19 -0
  86. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/contexts/business/index.tsx +12 -12
  87. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/contexts/color-mode/index.tsx +22 -26
  88. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/custom-avatar.tsx +13 -9
  89. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/index.ts +4 -4
  90. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/layout/header.tsx +22 -37
  91. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/layout/index.ts +4 -4
  92. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/layout/layout.tsx +10 -10
  93. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/layout/logo.tsx +1 -1
  94. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/layout/sider.tsx +113 -65
  95. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/layout/title.tsx +14 -14
  96. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/offLayoutArea/index.tsx +1 -1
  97. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/pagination-total.tsx +2 -2
  98. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/tags/index.ts +1 -1
  99. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/tags/role-tag.tsx +21 -21
  100. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/text.tsx +17 -13
  101. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/config/http.ts +26 -5
  102. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/config/routeRegistry.tsx +221 -51
  103. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/constants/layout.ts +3 -3
  104. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/dev-shell/AddDataModal.tsx +199 -0
  105. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/dev-shell/AddPageModal.tsx +335 -0
  106. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/dev-shell/DeleteDataConfirm.tsx +56 -0
  107. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/dev-shell/DevShell.tsx +401 -0
  108. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/dev-shell/EditDataModal.tsx +129 -0
  109. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/dev-shell/SkillMenu.tsx +202 -0
  110. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/dev-shell/UndoConfirm.tsx +36 -0
  111. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/dev-shell/index.tsx +8 -0
  112. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/dev-shell/style.css +835 -0
  113. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/enums/audit-log.enum.ts +8 -8
  114. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/enums/index.ts +1 -1
  115. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/helpers/form.tsx +120 -70
  116. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/helpers/get-name-initials.ts +3 -3
  117. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/helpers/get-random-color.ts +11 -11
  118. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/helpers/http.ts +16 -19
  119. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/helpers/index.tsx +1 -1
  120. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/helpers/list.tsx +99 -73
  121. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/helpers/login.ts +22 -27
  122. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/helpers/page-tree.ts +74 -0
  123. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/helpers/show.tsx +95 -51
  124. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/helpers/utils.tsx +7 -3
  125. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/hooks/types.ts +15 -15
  126. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/hooks/useBatchDelete.ts +10 -10
  127. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/hooks/useBatchOperations.ts +8 -7
  128. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/hooks/useBatchOperationsCore.ts +10 -10
  129. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/hooks/useBatchUpdate.ts +10 -10
  130. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/hooks/useDynamicResources.tsx +36 -27
  131. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/hooks/useECharts.tsx +144 -0
  132. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/hooks/useFetchData.ts +4 -4
  133. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/hooks/useListPageState.ts +34 -34
  134. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/hooks/useRoles.ts +8 -4
  135. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/hooks/useRowSelection.ts +1 -1
  136. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/hooks/useUserPageAccess.ts +55 -41
  137. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/i18n.ts +8 -8
  138. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/index.tsx +9 -2
  139. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/locales/en/common.json +193 -0
  140. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/locales/zh_CN/common.json +194 -0
  141. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/plugins/devShellPlugin.ts +81 -0
  142. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/audit/components/action-cell.tsx +37 -37
  143. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/audit/create.tsx +23 -31
  144. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/audit/edit.tsx +34 -35
  145. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/audit/index.ts +1 -1
  146. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/audit/index.tsx +4 -6
  147. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/audit/list.tsx +41 -34
  148. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/audit/show.tsx +33 -20
  149. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CanvasCell.tsx +87 -0
  150. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CanvasEditPage.tsx +192 -0
  151. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CanvasListPage.tsx +409 -0
  152. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CanvasPage.tsx +1372 -0
  153. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CanvasToolbar.tsx +384 -0
  154. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/CodeFloatWindow.tsx +476 -0
  155. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/ComponentThumbnail.tsx +135 -0
  156. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/IsolatedLivePreview.tsx +101 -0
  157. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/LivePreview.tsx +236 -0
  158. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/PublishModal.tsx +200 -0
  159. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/canvasApi.ts +92 -0
  160. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/canvasConfigRegistry.tsx +25 -0
  161. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/canvasContextMenuRegistry.tsx +875 -0
  162. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/canvasDefaults.ts +126 -0
  163. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/canvasI18n.ts +11 -0
  164. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/components/BarChartDataSourceModal.tsx +115 -0
  165. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/components/ChartViewerConfigModal.tsx +217 -0
  166. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/components/LineChartDataSourceModal.tsx +153 -0
  167. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/components/NumCardDataSourceModal.tsx +211 -0
  168. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/components/PromptModal.tsx +90 -0
  169. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/components/RadarChartDataSourceModal.tsx +87 -0
  170. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/components/TableDataSourceModal.tsx +204 -0
  171. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/demos.ts +1153 -0
  172. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/index.tsx +2 -0
  173. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/sectionCompactor.ts +45 -0
  174. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas/types.ts +45 -0
  175. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/canvas-page/index.tsx +108 -0
  176. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/game/create.tsx +105 -0
  177. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/game/edit.tsx +121 -0
  178. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/game/index.tsx +4 -0
  179. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/game/list.tsx +237 -0
  180. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/game/show.tsx +74 -0
  181. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/page/Components/AssignRolesModal.tsx +29 -20
  182. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/page/Components/CreatePageModal.tsx +3 -3
  183. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/page/Components/EditPageModal.tsx +7 -3
  184. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/page/Components/PageDetailDrawer.tsx +58 -23
  185. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/page/Components/PageFormModal.tsx +206 -122
  186. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/page/hooks/usePageManagement.ts +8 -4
  187. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/page/index.ts +1 -1
  188. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/page/list.tsx +345 -190
  189. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/page/types.ts +1 -1
  190. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/permissionReadme/index.tsx +341 -340
  191. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/resource/Components/CreateModal.tsx +2 -2
  192. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/resource/Components/EditModal.tsx +2 -2
  193. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/resource/Components/ResourceDetailDrawer.tsx +41 -24
  194. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/resource/Components/modal.tsx +64 -36
  195. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/resource/index.ts +2 -2
  196. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/resource/list.tsx +78 -40
  197. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/role/Components/CreateModal.tsx +3 -3
  198. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/role/Components/EditModal.tsx +4 -4
  199. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/role/Components/RoleDetailDrawer.tsx +19 -9
  200. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/role/Components/modal.tsx +45 -47
  201. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/role/hooks/useRolePage.ts +4 -4
  202. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/role/index.ts +1 -1
  203. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/role/list.tsx +121 -147
  204. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/role/types.ts +1 -1
  205. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/user/components/create-modal.tsx +2 -2
  206. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/user/components/edit-modal.tsx +2 -2
  207. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/user/components/form-modal.tsx +25 -19
  208. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/user/components/index.ts +5 -5
  209. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/user/components/role-tag.tsx +10 -10
  210. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/user/components/show-drawer.tsx +31 -26
  211. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/user/index.ts +1 -1
  212. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/routes/user/list.tsx +76 -71
  213. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/setupTests.ts +1 -1
  214. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/types/audit-log.ts +1 -1
  215. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/types/index.ts +3 -3
  216. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/types/role.ts +1 -1
  217. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/types/user.ts +1 -1
  218. package/dist/lib/application/files/gadmin2-game-angle-demo/web/vite.config.ts +42 -25
  219. package/dist/lib/application/files/gadmin2-game-angle-demo/web/yarn.lock +1193 -32
  220. package/package.json +1 -1
  221. package/dist/lib/application/files/gadmin2-game-angle-demo/server/seed/seedDataMngtPages.ts +0 -258
  222. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/AgentPanel.tsx +0 -497
  223. package/dist/lib/application/files/gadmin2-game-angle-demo/web/src/components/agentPanel/skills.ts +0 -464
@@ -0,0 +1,875 @@
1
+ import type { MenuProps } from 'antd';
2
+ import type { TFunction } from 'i18next';
3
+ import { PlusOutlined, MinusOutlined, EditOutlined } from '@ant-design/icons';
4
+
5
+ // ─── 组件特有右键菜单操作注册表 ────────────────────────────────────────────────
6
+ //
7
+ // 每种组件类型可以注册自己特有的右键菜单操作(第 1 层)。
8
+ // 操作通过直接修改组件的 code 字符串来实现。
9
+
10
+ export interface ContextMenuAction {
11
+ key: string;
12
+ label: string | ((code: string) => string);
13
+ icon?: React.ReactNode;
14
+ /** 执行操作,返回修改后的 code */
15
+ execute: (code: string) => string;
16
+ /** 可选:是否禁用 */
17
+ disabled?: (code: string) => boolean;
18
+ }
19
+
20
+ export type ContextMenuActionsFactory = (code: string) => MenuProps['items'];
21
+
22
+ /** 右键菜单 factory 的上下文,由 CanvasPage 提供 */
23
+ export interface MenuActionContext {
24
+ /** 打开文本输入弹窗,用户确认后回调 onConfirm */
25
+ showPrompt: (opts: {
26
+ label: string;
27
+ placeholder?: string;
28
+ defaultValue?: string;
29
+ onConfirm: (value: string) => void;
30
+ }) => void;
31
+ }
32
+
33
+ // ─── NumCard 特有操作 ─────────────────────────────────────────────────────────
34
+
35
+ function numCardActions(
36
+ code: string,
37
+ updateCode: (newCode: string) => void,
38
+ closeMenu: () => void,
39
+ ctx?: MenuActionContext,
40
+ t?: TFunction,
41
+ ): MenuProps['items'] {
42
+ const hasComparisons = /comparisons\s*=\s*\{/.test(code);
43
+ const comparisonsHidden = /showComparisons\s*=\s*\{false\}/.test(code);
44
+ const hasWithChartVariant = /variant\s*=\s*"withChart"/.test(code);
45
+ const chartHidden = /showChart\s*=\s*\{false\}/.test(code);
46
+ const isWithChart = hasWithChartVariant && !chartHidden;
47
+ // 如果有 comparisons 数据且没隐藏,或者没数据,都算"当前显示"状态用于 label
48
+ const comparisonsVisible = hasComparisons && !comparisonsHidden;
49
+
50
+ return [
51
+ ...chartTitleActions(code, updateCode, closeMenu, ctx, t)!,
52
+ {
53
+ key: 'toggle-comparisons',
54
+ icon: comparisonsVisible ? <MinusOutlined /> : <PlusOutlined />,
55
+ danger: comparisonsVisible ? true : undefined,
56
+ label: comparisonsVisible
57
+ ? (t?.('canvas.menu.removeComparisons') ?? '隐藏环比数据')
58
+ : (t?.('canvas.menu.addComparisons') ?? '显示环比/对比数据'),
59
+ onClick: () => {
60
+ let newCode = code;
61
+ if (comparisonsVisible) {
62
+ // 隐藏:注入 showComparisons={false},不删数据
63
+ if (/showComparisons\s*=/.test(newCode)) {
64
+ newCode = newCode.replace(
65
+ /showComparisons\s*=\s*\{true\}/,
66
+ 'showComparisons={false}',
67
+ );
68
+ } else {
69
+ if (/testId\s*=/.test(newCode)) {
70
+ newCode = newCode.replace(
71
+ /([ \t]*)(testId\s*=)/,
72
+ `$1showComparisons={false}\n$1$2`,
73
+ );
74
+ } else {
75
+ newCode = newCode.replace(/\/>/, ` showComparisons={false}\n/>`);
76
+ }
77
+ }
78
+ } else if (hasComparisons && comparisonsHidden) {
79
+ // 有数据但被隐藏了 → 显示:移除 showComparisons={false}
80
+ newCode = newCode.replace(/\s*showComparisons\s*=\s*\{false\}/, '');
81
+ } else {
82
+ // 没有 comparisons 数据 → 注入默认模板
83
+ const compTemplate = `comparisons={[\n { label: "比昨日", value: "0%", isUp: true },\n { label: "比上月", value: "0%", isUp: false },\n ]}`;
84
+ if (/testId\s*=/.test(newCode)) {
85
+ newCode = newCode.replace(
86
+ /([ \t]*)(testId\s*=)/,
87
+ `$1${compTemplate}\n$1$2`,
88
+ );
89
+ } else {
90
+ newCode = newCode.replace(/\/>/, ` ${compTemplate}\n/>`);
91
+ }
92
+ }
93
+ updateCode(newCode);
94
+ closeMenu();
95
+ },
96
+ },
97
+ {
98
+ key: 'toggle-chart',
99
+ icon: isWithChart ? <MinusOutlined /> : <PlusOutlined />,
100
+ danger: isWithChart ? true : undefined,
101
+ label: isWithChart
102
+ ? (t?.('canvas.menu.removeChart') ?? '隐藏右侧折线图')
103
+ : (t?.('canvas.menu.addChart') ?? '显示右侧折线图'),
104
+ onClick: () => {
105
+ let newCode = code;
106
+ if (isWithChart) {
107
+ // 隐藏折线图:注入 showChart={false},保留 variant 和数据
108
+ if (/showChart\s*=/.test(newCode)) {
109
+ newCode = newCode.replace(
110
+ /showChart\s*=\s*\{true\}/,
111
+ 'showChart={false}',
112
+ );
113
+ } else {
114
+ if (/testId\s*=/.test(newCode)) {
115
+ newCode = newCode.replace(
116
+ /([ \t]*)(testId\s*=)/,
117
+ `$1showChart={false}\n$1$2`,
118
+ );
119
+ } else {
120
+ newCode = newCode.replace(/\/>/, ` showChart={false}\n/>`);
121
+ }
122
+ }
123
+ } else {
124
+ // 显示折线图
125
+ const chartHidden = /showChart\s*=\s*\{false\}/.test(newCode);
126
+ if (chartHidden) {
127
+ // 有数据但被隐藏了 → 移除 showChart={false}
128
+ newCode = newCode.replace(/\s*showChart\s*=\s*\{false\}/, '');
129
+ } else {
130
+ // 从没有 chart → 切 variant + 注入默认数据
131
+ if (/variant\s*=\s*"[^"]*"/.test(newCode)) {
132
+ newCode = newCode.replace(
133
+ /variant\s*=\s*"[^"]*"/,
134
+ 'variant="withChart"',
135
+ );
136
+ } else {
137
+ newCode = newCode.replace(
138
+ /(<NumCard)/,
139
+ '$1\n variant="withChart"',
140
+ );
141
+ }
142
+ // 只有没有 chartData 时才注入默认模板
143
+ if (!/chartData\s*=/.test(newCode)) {
144
+ const chartTemplate = `chartData={[\n { name: "W1", value: 0 },\n { name: "W2", value: 0 },\n { name: "W3", value: 0 },\n { name: "W4", value: 0 },\n ]}\n chartTestId="canvas-numcard-chart"`;
145
+ if (/testId\s*=/.test(newCode)) {
146
+ newCode = newCode.replace(
147
+ /([ \t]*)(testId\s*=)/,
148
+ `$1${chartTemplate}\n$1$2`,
149
+ );
150
+ } else {
151
+ newCode = newCode.replace(/\/>/, ` ${chartTemplate}\n/>`);
152
+ }
153
+ }
154
+ }
155
+ }
156
+ updateCode(newCode);
157
+ closeMenu();
158
+ },
159
+ },
160
+ // 仅在折线图可见时显示坐标轴选项
161
+ ...(isWithChart
162
+ ? [
163
+ {
164
+ key: 'toggle-axes',
165
+ icon: /showAxes\s*=\s*\{true\}/.test(code) ? (
166
+ <MinusOutlined />
167
+ ) : (
168
+ <PlusOutlined />
169
+ ),
170
+ danger: /showAxes\s*=\s*\{true\}/.test(code) ? true : undefined,
171
+ label: /showAxes\s*=\s*\{true\}/.test(code)
172
+ ? (t?.('canvas.menu.hideAxes') ?? '隐藏坐标轴')
173
+ : (t?.('canvas.menu.showAxes') ?? '显示坐标轴'),
174
+ onClick: () => {
175
+ let newCode = code;
176
+ const hasAxes = /showAxes\s*=\s*\{true\}/.test(newCode);
177
+ if (hasAxes) {
178
+ newCode = newCode.replace(/\s*showAxes\s*=\s*\{true\}/, '');
179
+ } else {
180
+ if (/testId\s*=/.test(newCode)) {
181
+ newCode = newCode.replace(
182
+ /([ \t]*)(testId\s*=)/,
183
+ `$1showAxes={true}\n$1$2`,
184
+ );
185
+ } else {
186
+ newCode = newCode.replace(/\/>/, ` showAxes={true}\n/>`);
187
+ }
188
+ }
189
+ updateCode(newCode);
190
+ closeMenu();
191
+ },
192
+ },
193
+ ]
194
+ : []),
195
+ ];
196
+ }
197
+
198
+ // ─── Table 特有操作 ──────────────────────────────────────────────────────────
199
+
200
+ const PAGE_SIZE_OPTIONS = [10, 20, 50, 100];
201
+
202
+ function tableActions(
203
+ code: string,
204
+ updateCode: (newCode: string) => void,
205
+ closeMenu: () => void,
206
+ ctx?: MenuActionContext,
207
+ t?: TFunction,
208
+ ): MenuProps['items'] {
209
+ const hasPagination = /pagination\s*=\s*\{\{/.test(code);
210
+
211
+ if (hasPagination) {
212
+ // 已有分页:显示"修改分页数量"子菜单 + "移除分页"
213
+ const currentPageSize = code.match(/pageSize\s*:\s*(\d+)/)?.[1] ?? '10';
214
+
215
+ return [
216
+ {
217
+ key: 'change-page-size',
218
+ icon: <PlusOutlined />,
219
+ label: t?.('canvas.menu.changePageSize') ?? '修改分页数量',
220
+ children: PAGE_SIZE_OPTIONS.map((size) => ({
221
+ key: `page-size-${size}`,
222
+ label: `${size} 条/页${String(size) === currentPageSize ? ' ✓' : ''}`,
223
+ onClick: () => {
224
+ const newCode = code.replace(
225
+ /pageSize\s*:\s*\d+/,
226
+ `pageSize: ${size}`,
227
+ );
228
+ updateCode(newCode);
229
+ closeMenu();
230
+ },
231
+ })),
232
+ },
233
+ {
234
+ key: 'remove-pagination',
235
+ icon: <MinusOutlined />,
236
+ label: t?.('canvas.menu.removePagination') ?? '移除分页',
237
+ danger: true,
238
+ onClick: () => {
239
+ let newCode = code;
240
+ newCode = newCode.replace(/\s*pagination\s*=\s*\{\{[\s\S]*?\}\}/, '');
241
+ newCode = newCode.replace(/\s*entityName\s*=\s*"[^"]*"/, '');
242
+ updateCode(newCode);
243
+ closeMenu();
244
+ },
245
+ },
246
+ ];
247
+ }
248
+
249
+ // 无分页:显示"添加分页"子菜单,选择 pageSize
250
+ return [
251
+ {
252
+ key: 'add-pagination',
253
+ icon: <PlusOutlined />,
254
+ label: t?.('canvas.menu.addPagination') ?? '添加分页',
255
+ children: PAGE_SIZE_OPTIONS.map((size) => ({
256
+ key: `add-page-size-${size}`,
257
+ label: `${size} 条/页`,
258
+ onClick: () => {
259
+ let newCode = code;
260
+ const paginationTemplate = `pagination={{ current: 1, pageSize: ${size}, total: 0 }}`;
261
+ if (/testId\s*=/.test(newCode)) {
262
+ newCode = newCode.replace(
263
+ /([ \t]*)(testId\s*=)/,
264
+ `$1${paginationTemplate}\n$1$2`,
265
+ );
266
+ } else {
267
+ newCode = newCode.replace(/\/>/, ` ${paginationTemplate}\n/>`);
268
+ }
269
+ updateCode(newCode);
270
+ closeMenu();
271
+ },
272
+ })),
273
+ },
274
+ ];
275
+ }
276
+
277
+ // ─── 通用 Chart 操作:图例显隐 ─────────────────────────────────────────────────
278
+
279
+ function chartLegendActions(
280
+ code: string,
281
+ updateCode: (newCode: string) => void,
282
+ closeMenu: () => void,
283
+ t?: TFunction,
284
+ ): MenuProps['items'] {
285
+ const hasLegendShow = /legendShow\s*=\s*\{/.test(code);
286
+ const legendIsHidden = /legendShow\s*=\s*\{false\}/.test(code);
287
+
288
+ // 当前状态:没有显式设置 legendShow 或 legendShow={true} → 图例可见
289
+ const isVisible = !hasLegendShow || !legendIsHidden;
290
+
291
+ return [
292
+ {
293
+ key: 'toggle-legend',
294
+ icon: isVisible ? <MinusOutlined /> : <PlusOutlined />,
295
+ label: isVisible
296
+ ? (t?.('canvas.menu.hideLegend') ?? '隐藏图例')
297
+ : (t?.('canvas.menu.showLegend') ?? '显示图例'),
298
+ danger: isVisible ? true : undefined,
299
+ onClick: () => {
300
+ let newCode = code;
301
+ if (isVisible) {
302
+ // 当前可见 → 隐藏
303
+ if (hasLegendShow) {
304
+ newCode = newCode.replace(
305
+ /legendShow\s*=\s*\{true\}/,
306
+ 'legendShow={false}',
307
+ );
308
+ } else {
309
+ // 注入 legendShow={false}
310
+ if (/testId\s*=/.test(newCode)) {
311
+ newCode = newCode.replace(
312
+ /([ \t]*)(testId\s*=)/,
313
+ `$1legendShow={false}\n$1$2`,
314
+ );
315
+ } else {
316
+ newCode = newCode.replace(/\/>/, ` legendShow={false}\n/>`);
317
+ }
318
+ }
319
+ } else {
320
+ // 当前隐藏 → 显示:将 legendShow={false} 替换为 legendShow={true}
321
+ newCode = newCode.replace(
322
+ /legendShow\s*=\s*\{false\}/,
323
+ 'legendShow={true}',
324
+ );
325
+ }
326
+ updateCode(newCode);
327
+ closeMenu();
328
+ },
329
+ },
330
+ ];
331
+ }
332
+
333
+ // ─── 通用 Chart 操作:修改标题 ─────────────────────────────────────────────────
334
+
335
+ function chartTitleActions(
336
+ code: string,
337
+ updateCode: (newCode: string) => void,
338
+ closeMenu: () => void,
339
+ ctx?: MenuActionContext,
340
+ t?: TFunction,
341
+ ): MenuProps['items'] {
342
+ const hasTitle = /title\s*=\s*"/.test(code);
343
+ const currentTitle = code.match(/title\s*=\s*"([^"]*)"/)?.[1] ?? '';
344
+
345
+ return [
346
+ {
347
+ key: 'edit-title',
348
+ icon: <EditOutlined />,
349
+ label: hasTitle
350
+ ? (t?.('canvas.menu.editTitle') ?? '修改标题')
351
+ : (t?.('canvas.menu.addTitle') ?? '添加标题'),
352
+ onClick: () => {
353
+ closeMenu();
354
+ ctx?.showPrompt({
355
+ label: hasTitle
356
+ ? (t?.('canvas.menu.editChartTitle') ?? '修改图表标题')
357
+ : (t?.('canvas.menu.addChartTitle') ?? '添加图表标题'),
358
+ placeholder:
359
+ t?.('canvas.menu.titlePlaceholder') ?? '请输入标题(留空则移除)',
360
+ defaultValue: currentTitle,
361
+ onConfirm: (newTitle) => {
362
+ let newCode = code;
363
+ if (newTitle === '') {
364
+ // 移除标题
365
+ newCode = newCode.replace(/\s*title\s*=\s*"[^"]*"/, '');
366
+ } else if (hasTitle) {
367
+ // 修改标题
368
+ newCode = newCode.replace(
369
+ /title\s*=\s*"[^"]*"/,
370
+ `title="${newTitle}"`,
371
+ );
372
+ } else {
373
+ // 注入标题
374
+ if (/testId\s*=/.test(newCode)) {
375
+ newCode = newCode.replace(
376
+ /([ \t]*)(testId\s*=)/,
377
+ `$1title="${newTitle}"\n$1$2`,
378
+ );
379
+ } else {
380
+ newCode = newCode.replace(/\/>/, ` title="${newTitle}"\n/>`);
381
+ }
382
+ }
383
+ updateCode(newCode);
384
+ },
385
+ });
386
+ },
387
+ },
388
+ ];
389
+ }
390
+
391
+ // ─── BarChart 特有操作 ─────────────────────────────────────────────────────
392
+
393
+ function barChartActions(
394
+ code: string,
395
+ updateCode: (newCode: string) => void,
396
+ closeMenu: () => void,
397
+ ctx?: MenuActionContext,
398
+ t?: TFunction,
399
+ ): MenuProps['items'] {
400
+ const isHorizontal = /direction\s*=\s*"horizontal"/.test(code);
401
+ const gridIsHidden = /showGrid\s*=\s*\{false\}/.test(code);
402
+ const labelIsHidden = /showLabel\s*=\s*\{false\}/.test(code);
403
+
404
+ return [
405
+ {
406
+ key: 'bar-direction',
407
+ icon: <PlusOutlined />,
408
+ label: t?.('canvas.menu.direction') ?? '方向',
409
+ children: [
410
+ {
411
+ key: 'direction-vertical',
412
+ label: `${t?.('canvas.menu.vertical') ?? '纵向'}${!isHorizontal ? ' ✓' : ''}`,
413
+ onClick: () => {
414
+ let newCode = code;
415
+ // 移除 direction prop(vertical 是默认值,不需要显式写)
416
+ newCode = newCode.replace(/\s*direction\s*=\s*"horizontal"/, '');
417
+ updateCode(newCode);
418
+ closeMenu();
419
+ },
420
+ },
421
+ {
422
+ key: 'direction-horizontal',
423
+ label: `${t?.('canvas.menu.horizontal') ?? '横向'}${isHorizontal ? ' ✓' : ''}`,
424
+ onClick: () => {
425
+ let newCode = code;
426
+ if (!isHorizontal) {
427
+ // 注入 direction="horizontal" 到 testId 前
428
+ if (/testId\s*=/.test(newCode)) {
429
+ newCode = newCode.replace(
430
+ /([ \t]*)(testId\s*=)/,
431
+ `$1direction="horizontal"\n$1$2`,
432
+ );
433
+ } else {
434
+ newCode = newCode.replace(
435
+ /\/>/,
436
+ ` direction="horizontal"\n/>`,
437
+ );
438
+ }
439
+ }
440
+ updateCode(newCode);
441
+ closeMenu();
442
+ },
443
+ },
444
+ ],
445
+ },
446
+ {
447
+ key: 'toggle-grid',
448
+ icon: gridIsHidden ? <PlusOutlined /> : <MinusOutlined />,
449
+ label: gridIsHidden
450
+ ? (t?.('canvas.menu.showGrid') ?? '显示网格')
451
+ : (t?.('canvas.menu.hideGrid') ?? '隐藏网格'),
452
+ danger: !gridIsHidden ? true : undefined,
453
+ onClick: () => {
454
+ let newCode = code;
455
+ if (gridIsHidden) {
456
+ newCode = newCode.replace(/\s*showGrid\s*=\s*\{false\}/, '');
457
+ } else {
458
+ if (/testId\s*=/.test(newCode)) {
459
+ newCode = newCode.replace(
460
+ /([ \t]*)(testId\s*=)/,
461
+ `$1showGrid={false}\n$1$2`,
462
+ );
463
+ } else {
464
+ newCode = newCode.replace(/\/>/, ` showGrid={false}\n/>`);
465
+ }
466
+ }
467
+ updateCode(newCode);
468
+ closeMenu();
469
+ },
470
+ },
471
+ {
472
+ key: 'toggle-label',
473
+ icon: labelIsHidden ? <PlusOutlined /> : <MinusOutlined />,
474
+ label: labelIsHidden
475
+ ? (t?.('canvas.menu.showLabel') ?? '显示数据标签')
476
+ : (t?.('canvas.menu.hideLabel') ?? '隐藏数据标签'),
477
+ danger: !labelIsHidden ? true : undefined,
478
+ onClick: () => {
479
+ let newCode = code;
480
+ if (labelIsHidden) {
481
+ newCode = newCode.replace(/\s*showLabel\s*=\s*\{false\}/, '');
482
+ } else {
483
+ if (/testId\s*=/.test(newCode)) {
484
+ newCode = newCode.replace(
485
+ /([ \t]*)(testId\s*=)/,
486
+ `$1showLabel={false}\n$1$2`,
487
+ );
488
+ } else {
489
+ newCode = newCode.replace(/\/>/, ` showLabel={false}\n/>`);
490
+ }
491
+ }
492
+ updateCode(newCode);
493
+ closeMenu();
494
+ },
495
+ },
496
+ ...chartLegendActions(code, updateCode, closeMenu, t)!,
497
+ ...chartTitleActions(code, updateCode, closeMenu, ctx, t)!,
498
+ ];
499
+ }
500
+
501
+ // ─── PieChart 特有操作 ─────────────────────────────────────────────────────
502
+
503
+ function pieChartActions(
504
+ code: string,
505
+ updateCode: (newCode: string) => void,
506
+ closeMenu: () => void,
507
+ ctx?: MenuActionContext,
508
+ t?: TFunction,
509
+ ): MenuProps['items'] {
510
+ const currentVariant =
511
+ code.match(/variant\s*=\s*"(pie|ring|sunburst)"/)?.[1] ?? 'pie';
512
+ const isSunburst = currentVariant === 'sunburst';
513
+
514
+ const items: MenuProps['items'] = [];
515
+
516
+ // variant 切换(sunburst 不可切换)
517
+ if (!isSunburst) {
518
+ items.push({
519
+ key: 'pie-variant',
520
+ icon: <PlusOutlined />,
521
+ label: t?.('canvas.menu.switchVariant') ?? '切换样式',
522
+ children: [
523
+ {
524
+ key: 'variant-pie',
525
+ label: `${t?.('canvas.menu.pie') ?? '饼图'}${currentVariant === 'pie' ? ' ✓' : ''}`,
526
+ onClick: () => {
527
+ const newCode = code.replace(
528
+ /variant\s*=\s*"(pie|ring)"/,
529
+ 'variant="pie"',
530
+ );
531
+ updateCode(newCode);
532
+ closeMenu();
533
+ },
534
+ },
535
+ {
536
+ key: 'variant-ring',
537
+ label: `${t?.('canvas.menu.ring') ?? '环形图'}${currentVariant === 'ring' ? ' ✓' : ''}`,
538
+ onClick: () => {
539
+ const newCode = code.replace(
540
+ /variant\s*=\s*"(pie|ring)"/,
541
+ 'variant="ring"',
542
+ );
543
+ updateCode(newCode);
544
+ closeMenu();
545
+ },
546
+ },
547
+ ],
548
+ });
549
+
550
+ // topN 显示项数(仅 pie/ring 生效)
551
+ const topNMatch = code.match(/topN\s*=\s*\{(\d+)\}/);
552
+ const currentTopN = topNMatch ? Number(topNMatch[1]) : 5; // 默认 5
553
+ const TOP_N_OPTIONS = [3, 5, 8, 10, 0]; // 0 = 不限制
554
+
555
+ items.push({
556
+ key: 'pie-topn',
557
+ icon: <PlusOutlined />,
558
+ label: t?.('canvas.menu.topN') ?? '显示项数',
559
+ children: TOP_N_OPTIONS.map((n) => ({
560
+ key: `topn-${n}`,
561
+ label: `${n === 0 ? '不限制' : `${n} 项`}${currentTopN === n ? ' ✓' : ''}`,
562
+ onClick: () => {
563
+ let newCode = code;
564
+ if (topNMatch) {
565
+ // 已有 topN → 替换
566
+ newCode = newCode.replace(/topN\s*=\s*\{\d+\}/, `topN={${n}}`);
567
+ } else {
568
+ // 无 topN → 注入
569
+ if (/testId\s*=/.test(newCode)) {
570
+ newCode = newCode.replace(
571
+ /([ \t]*)(testId\s*=)/,
572
+ `$1topN={${n}}\n$1$2`,
573
+ );
574
+ } else {
575
+ newCode = newCode.replace(/\/>/, ` topN={${n}}\n/>`);
576
+ }
577
+ }
578
+ updateCode(newCode);
579
+ closeMenu();
580
+ },
581
+ })),
582
+ });
583
+ }
584
+
585
+ // 图例 + 标题(所有 chart 通用)
586
+ items.push(...chartLegendActions(code, updateCode, closeMenu, t)!);
587
+ items.push(...chartTitleActions(code, updateCode, closeMenu, ctx, t)!);
588
+
589
+ return items;
590
+ }
591
+
592
+ // ─── LineChart 特有操作 ─────────────────────────────────────────────────────
593
+
594
+ function lineChartActions(
595
+ code: string,
596
+ updateCode: (newCode: string) => void,
597
+ closeMenu: () => void,
598
+ ctx?: MenuActionContext,
599
+ t?: TFunction,
600
+ ): MenuProps['items'] {
601
+ // ── showGrid toggle(默认 true,隐藏 = 注入 {false})──
602
+ const gridIsHidden = /showGrid\s*=\s*\{false\}/.test(code);
603
+
604
+ // ── showDots toggle(默认 false,显示 = 注入 {true})──
605
+ const dotsIsShown = /showDots\s*=\s*\{true\}/.test(code);
606
+
607
+ // ── showLabel toggle(默认 false,显示 = 注入 {true},仅 showDots 时有意义)──
608
+ const labelIsShown = /showLabel\s*=\s*\{true\}/.test(code);
609
+
610
+ // ── stacked toggle(默认 false,开启 = 注入 {true})──
611
+ const isStacked = /stacked\s*=\s*\{true\}/.test(code);
612
+
613
+ // ── area toggle(默认 false,开启 = 注入 {true})──
614
+ const isArea = /area\s*=\s*\{true\}/.test(code);
615
+
616
+ // ── smooth toggle(默认 false,开启 = 注入 {true})──
617
+ const isSmooth =
618
+ /smooth\s*=\s*\{true\}/.test(code) || /\bsmooth\b(?!\s*=)/.test(code);
619
+
620
+ // ── showAxis toggle(默认 true,隐藏 = 注入 {false})──
621
+ const axisIsHidden = /showAxis\s*=\s*\{false\}/.test(code);
622
+
623
+ return [
624
+ ...chartLegendActions(code, updateCode, closeMenu, t)!,
625
+ {
626
+ key: 'toggle-grid',
627
+ icon: gridIsHidden ? <PlusOutlined /> : <MinusOutlined />,
628
+ label: gridIsHidden
629
+ ? (t?.('canvas.menu.showGrid') ?? '显示网格')
630
+ : (t?.('canvas.menu.hideGrid') ?? '隐藏网格'),
631
+ danger: !gridIsHidden ? true : undefined,
632
+ onClick: () => {
633
+ let newCode = code;
634
+ if (gridIsHidden) {
635
+ // 当前隐藏 → 显示:移除 showGrid={false}
636
+ newCode = newCode.replace(/\s*showGrid\s*=\s*\{false\}/, '');
637
+ } else {
638
+ // 当前显示 → 隐藏:注入 showGrid={false}
639
+ if (/testId\s*=/.test(newCode)) {
640
+ newCode = newCode.replace(
641
+ /([ \t]*)(testId\s*=)/,
642
+ `$1showGrid={false}\n$1$2`,
643
+ );
644
+ } else {
645
+ newCode = newCode.replace(/\/>/, ` showGrid={false}\n/>`);
646
+ }
647
+ }
648
+ updateCode(newCode);
649
+ closeMenu();
650
+ },
651
+ },
652
+ {
653
+ key: 'toggle-axis',
654
+ icon: axisIsHidden ? <PlusOutlined /> : <MinusOutlined />,
655
+ label: axisIsHidden
656
+ ? (t?.('canvas.menu.showAxis') ?? '显示坐标轴')
657
+ : (t?.('canvas.menu.hideAxis') ?? '隐藏坐标轴'),
658
+ danger: !axisIsHidden ? true : undefined,
659
+ onClick: () => {
660
+ let newCode = code;
661
+ if (axisIsHidden) {
662
+ // 当前隐藏 → 显示:移除 showAxis={false}
663
+ newCode = newCode.replace(/\s*showAxis\s*=\s*\{false\}/, '');
664
+ } else {
665
+ // 当前显示 → 隐藏:注入 showAxis={false}
666
+ if (/testId\s*=/.test(newCode)) {
667
+ newCode = newCode.replace(
668
+ /([ \t]*)(testId\s*=)/,
669
+ `$1showAxis={false}\n$1$2`,
670
+ );
671
+ } else {
672
+ newCode = newCode.replace(/\/>/, ` showAxis={false}\n/>`);
673
+ }
674
+ }
675
+ updateCode(newCode);
676
+ closeMenu();
677
+ },
678
+ },
679
+ {
680
+ key: 'toggle-dots',
681
+ icon: dotsIsShown ? <MinusOutlined /> : <PlusOutlined />,
682
+ label: dotsIsShown
683
+ ? (t?.('canvas.menu.hideDots') ?? '隐藏坐标点')
684
+ : (t?.('canvas.menu.showDots') ?? '显示坐标点'),
685
+ danger: dotsIsShown ? true : undefined,
686
+ onClick: () => {
687
+ let newCode = code;
688
+ if (dotsIsShown) {
689
+ // 当前显示 → 隐藏:移除 showDots={true}
690
+ newCode = newCode.replace(/\s*showDots\s*=\s*\{true\}/, '');
691
+ } else {
692
+ // 当前隐藏 → 显示:注入 showDots={true}
693
+ if (/testId\s*=/.test(newCode)) {
694
+ newCode = newCode.replace(
695
+ /([ \t]*)(testId\s*=)/,
696
+ `$1showDots={true}\n$1$2`,
697
+ );
698
+ } else {
699
+ newCode = newCode.replace(/\/>/, ` showDots={true}\n/>`);
700
+ }
701
+ }
702
+ updateCode(newCode);
703
+ closeMenu();
704
+ },
705
+ },
706
+ // showLabel 仅在 showDots 开启时显示
707
+ ...(dotsIsShown
708
+ ? [
709
+ {
710
+ key: 'toggle-label',
711
+ icon: labelIsShown ? <MinusOutlined /> : <PlusOutlined />,
712
+ label: labelIsShown
713
+ ? (t?.('canvas.menu.hideDotsLabel') ?? '隐藏坐标点数据')
714
+ : (t?.('canvas.menu.showDotsLabel') ?? '显示坐标点数据'),
715
+ danger: labelIsShown ? true : undefined,
716
+ onClick: () => {
717
+ let newCode = code;
718
+ if (labelIsShown) {
719
+ // 隐藏:移除 showLabel={true}
720
+ newCode = newCode.replace(/\s*showLabel\s*=\s*\{true\}/, '');
721
+ } else {
722
+ // 显示:注入 showLabel={true}
723
+ if (/testId\s*=/.test(newCode)) {
724
+ newCode = newCode.replace(
725
+ /([ \t]*)(testId\s*=)/,
726
+ `$1showLabel={true}\n$1$2`,
727
+ );
728
+ } else {
729
+ newCode = newCode.replace(/\/>/, ` showLabel={true}\n/>`);
730
+ }
731
+ }
732
+ updateCode(newCode);
733
+ closeMenu();
734
+ },
735
+ },
736
+ ]
737
+ : []),
738
+ {
739
+ key: 'toggle-stacked',
740
+ icon: isStacked ? <MinusOutlined /> : <PlusOutlined />,
741
+ label: isStacked
742
+ ? (t?.('canvas.menu.disableStacked') ?? '关闭堆叠')
743
+ : (t?.('canvas.menu.enableStacked') ?? '开启堆叠'),
744
+ danger: isStacked ? true : undefined,
745
+ onClick: () => {
746
+ let newCode = code;
747
+ if (isStacked) {
748
+ // 关闭:移除 stacked={true}
749
+ newCode = newCode.replace(/\s*stacked\s*=\s*\{true\}/, '');
750
+ } else {
751
+ // 开启:注入 stacked={true}
752
+ if (/testId\s*=/.test(newCode)) {
753
+ newCode = newCode.replace(
754
+ /([ \t]*)(testId\s*=)/,
755
+ `$1stacked={true}\n$1$2`,
756
+ );
757
+ } else {
758
+ newCode = newCode.replace(/\/>/, ` stacked={true}\n/>`);
759
+ }
760
+ }
761
+ updateCode(newCode);
762
+ closeMenu();
763
+ },
764
+ },
765
+ {
766
+ key: 'toggle-area',
767
+ icon: isArea ? <MinusOutlined /> : <PlusOutlined />,
768
+ label: isArea
769
+ ? (t?.('canvas.menu.disableArea') ?? '关闭面积')
770
+ : (t?.('canvas.menu.enableArea') ?? '开启面积'),
771
+ danger: isArea ? true : undefined,
772
+ onClick: () => {
773
+ let newCode = code;
774
+ if (isArea) {
775
+ // 关闭:移除 area={true}
776
+ newCode = newCode.replace(/\s*area\s*=\s*\{true\}/, '');
777
+ } else {
778
+ // 开启:注入 area={true}
779
+ if (/testId\s*=/.test(newCode)) {
780
+ newCode = newCode.replace(
781
+ /([ \t]*)(testId\s*=)/,
782
+ `$1area={true}\n$1$2`,
783
+ );
784
+ } else {
785
+ newCode = newCode.replace(/\/>/, ` area={true}\n/>`);
786
+ }
787
+ }
788
+ updateCode(newCode);
789
+ closeMenu();
790
+ },
791
+ },
792
+ {
793
+ key: 'toggle-smooth',
794
+ icon: isSmooth ? <MinusOutlined /> : <PlusOutlined />,
795
+ label: isSmooth
796
+ ? (t?.('canvas.menu.disableSmooth') ?? '关闭拟合曲线')
797
+ : (t?.('canvas.menu.enableSmooth') ?? '开启拟合曲线'),
798
+ danger: isSmooth ? true : undefined,
799
+ onClick: () => {
800
+ let newCode = code;
801
+ if (isSmooth) {
802
+ // 关闭:移除 smooth={true} 或 smooth(布尔简写)
803
+ newCode = newCode.replace(/\s*smooth\s*=\s*\{true\}/, '');
804
+ newCode = newCode.replace(/\s*smooth\b(?!\s*=)/, '');
805
+ } else {
806
+ // 开启:注入 smooth={true}
807
+ if (/testId\s*=/.test(newCode)) {
808
+ newCode = newCode.replace(
809
+ /([ \t]*)(testId\s*=)/,
810
+ `$1smooth={true}\n$1$2`,
811
+ );
812
+ } else {
813
+ newCode = newCode.replace(/\/>/, ` smooth={true}\n/>`);
814
+ }
815
+ }
816
+ updateCode(newCode);
817
+ closeMenu();
818
+ },
819
+ },
820
+ ...chartTitleActions(code, updateCode, closeMenu, ctx, t)!,
821
+ ];
822
+ }
823
+
824
+ // ─── RadarChart 特有操作 ───────────────────────────────────────────────────
825
+
826
+ function radarChartActions(
827
+ code: string,
828
+ updateCode: (newCode: string) => void,
829
+ closeMenu: () => void,
830
+ ctx?: MenuActionContext,
831
+ t?: TFunction,
832
+ ): MenuProps['items'] {
833
+ return [
834
+ ...chartLegendActions(code, updateCode, closeMenu, t)!,
835
+ ...chartTitleActions(code, updateCode, closeMenu, ctx, t)!,
836
+ ];
837
+ }
838
+
839
+ // ─── TreemapChart 特有操作 ─────────────────────────────────────────────────
840
+
841
+ function treemapChartActions(
842
+ code: string,
843
+ updateCode: (newCode: string) => void,
844
+ closeMenu: () => void,
845
+ ctx?: MenuActionContext,
846
+ t?: TFunction,
847
+ ): MenuProps['items'] {
848
+ return [
849
+ ...chartLegendActions(code, updateCode, closeMenu, t)!,
850
+ ...chartTitleActions(code, updateCode, closeMenu, ctx, t)!,
851
+ ];
852
+ }
853
+
854
+ // ─── 注册表 ───────────────────────────────────────────────────────────────────
855
+
856
+ type ComponentActionsFactory = (
857
+ code: string,
858
+ updateCode: (newCode: string) => void,
859
+ closeMenu: () => void,
860
+ ctx?: MenuActionContext,
861
+ t?: TFunction,
862
+ ) => MenuProps['items'];
863
+
864
+ export const CANVAS_CONTEXT_MENU_REGISTRY: Record<
865
+ string,
866
+ ComponentActionsFactory
867
+ > = {
868
+ NumCard: numCardActions,
869
+ Table: tableActions,
870
+ BarChart: barChartActions,
871
+ PieChart: pieChartActions,
872
+ LineChart: lineChartActions,
873
+ RadarChart: radarChartActions,
874
+ TreemapChart: treemapChartActions,
875
+ };