create-aiko 0.1.0

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 (317) hide show
  1. package/README.md +46 -0
  2. package/dist/cli.d.ts +2 -0
  3. package/dist/cli.js +7 -0
  4. package/dist/create.d.ts +1 -0
  5. package/dist/create.js +174 -0
  6. package/dist/index.d.ts +2 -0
  7. package/dist/index.js +2 -0
  8. package/dist/scaffold.d.ts +7 -0
  9. package/dist/scaffold.js +310 -0
  10. package/package.json +37 -0
  11. package/template/README.md +63 -0
  12. package/template/docs/better-sqlite3-design.md +101 -0
  13. package/template/docs/shared-vs-shared-auth.md +54 -0
  14. package/template/package.json +23 -0
  15. package/template/packages/admin/.env.dev +1 -0
  16. package/template/packages/admin/.env.prod +4 -0
  17. package/template/packages/admin/.env.stage +4 -0
  18. package/template/packages/admin/Dockerfile +37 -0
  19. package/template/packages/admin/README.MD +27 -0
  20. package/template/packages/admin/components.json +21 -0
  21. package/template/packages/admin/eslint.config.js +28 -0
  22. package/template/packages/admin/index.html +50 -0
  23. package/template/packages/admin/package.json +100 -0
  24. package/template/packages/admin/public/vite.svg +1 -0
  25. package/template/packages/admin/src/App.css +82 -0
  26. package/template/packages/admin/src/App.tsx +128 -0
  27. package/template/packages/admin/src/app.config.ts +3 -0
  28. package/template/packages/admin/src/components/admin-ui/approval-flow.tsx +205 -0
  29. package/template/packages/admin/src/components/admin-ui/data-table/data-table-pagination.tsx +148 -0
  30. package/template/packages/admin/src/components/admin-ui/data-table/data-table-sorter.tsx +50 -0
  31. package/template/packages/admin/src/components/admin-ui/data-table/index.tsx +296 -0
  32. package/template/packages/admin/src/components/admin-ui/editable-table.tsx +292 -0
  33. package/template/packages/admin/src/components/admin-ui/form/input-password.tsx +38 -0
  34. package/template/packages/admin/src/components/admin-ui/form/sign-in-form.tsx +104 -0
  35. package/template/packages/admin/src/components/admin-ui/layout/error-component.tsx +71 -0
  36. package/template/packages/admin/src/components/admin-ui/layout/header.tsx +148 -0
  37. package/template/packages/admin/src/components/admin-ui/layout/language-switcher.tsx +47 -0
  38. package/template/packages/admin/src/components/admin-ui/layout/layout.tsx +42 -0
  39. package/template/packages/admin/src/components/admin-ui/layout/loading-overlay.tsx +36 -0
  40. package/template/packages/admin/src/components/admin-ui/layout/shell-bar.tsx +232 -0
  41. package/template/packages/admin/src/components/admin-ui/layout/sidebar.tsx +193 -0
  42. package/template/packages/admin/src/components/admin-ui/layout/user-avatar.tsx +31 -0
  43. package/template/packages/admin/src/components/admin-ui/list-report.tsx +313 -0
  44. package/template/packages/admin/src/components/admin-ui/master-detail.tsx +382 -0
  45. package/template/packages/admin/src/components/admin-ui/notification/toaster.tsx +23 -0
  46. package/template/packages/admin/src/components/admin-ui/notification/undoable-notification.tsx +84 -0
  47. package/template/packages/admin/src/components/admin-ui/object-page.tsx +539 -0
  48. package/template/packages/admin/src/components/admin-ui/process-stepper.tsx +204 -0
  49. package/template/packages/admin/src/components/admin-ui/theme/theme-provider.tsx +160 -0
  50. package/template/packages/admin/src/components/admin-ui/theme/theme-select.tsx +129 -0
  51. package/template/packages/admin/src/components/admin-ui/theme/theme-toggle.tsx +90 -0
  52. package/template/packages/admin/src/components/admin-ui/timeline.tsx +137 -0
  53. package/template/packages/admin/src/components/admin-ui/tree-navigator.tsx +243 -0
  54. package/template/packages/admin/src/components/ui/accordion.tsx +64 -0
  55. package/template/packages/admin/src/components/ui/alert-dialog.tsx +157 -0
  56. package/template/packages/admin/src/components/ui/alert.tsx +66 -0
  57. package/template/packages/admin/src/components/ui/aspect-ratio.tsx +9 -0
  58. package/template/packages/admin/src/components/ui/avatar.tsx +53 -0
  59. package/template/packages/admin/src/components/ui/badge.tsx +47 -0
  60. package/template/packages/admin/src/components/ui/breadcrumb.tsx +111 -0
  61. package/template/packages/admin/src/components/ui/button.tsx +59 -0
  62. package/template/packages/admin/src/components/ui/calendar.tsx +74 -0
  63. package/template/packages/admin/src/components/ui/card.tsx +92 -0
  64. package/template/packages/admin/src/components/ui/carousel.tsx +237 -0
  65. package/template/packages/admin/src/components/ui/chart.tsx +351 -0
  66. package/template/packages/admin/src/components/ui/checkbox.tsx +32 -0
  67. package/template/packages/admin/src/components/ui/collapsible.tsx +33 -0
  68. package/template/packages/admin/src/components/ui/command.tsx +182 -0
  69. package/template/packages/admin/src/components/ui/context-menu.tsx +252 -0
  70. package/template/packages/admin/src/components/ui/dialog.tsx +141 -0
  71. package/template/packages/admin/src/components/ui/drawer.tsx +130 -0
  72. package/template/packages/admin/src/components/ui/dropdown-menu.tsx +255 -0
  73. package/template/packages/admin/src/components/ui/form.tsx +166 -0
  74. package/template/packages/admin/src/components/ui/hover-card.tsx +42 -0
  75. package/template/packages/admin/src/components/ui/input-otp.tsx +77 -0
  76. package/template/packages/admin/src/components/ui/input.tsx +21 -0
  77. package/template/packages/admin/src/components/ui/label.tsx +22 -0
  78. package/template/packages/admin/src/components/ui/menubar.tsx +274 -0
  79. package/template/packages/admin/src/components/ui/navigation-menu.tsx +168 -0
  80. package/template/packages/admin/src/components/ui/pagination.tsx +127 -0
  81. package/template/packages/admin/src/components/ui/popover.tsx +48 -0
  82. package/template/packages/admin/src/components/ui/progress.tsx +29 -0
  83. package/template/packages/admin/src/components/ui/radio-group.tsx +45 -0
  84. package/template/packages/admin/src/components/ui/resizable.tsx +54 -0
  85. package/template/packages/admin/src/components/ui/scroll-area.tsx +58 -0
  86. package/template/packages/admin/src/components/ui/select.tsx +183 -0
  87. package/template/packages/admin/src/components/ui/separator.tsx +26 -0
  88. package/template/packages/admin/src/components/ui/sheet.tsx +139 -0
  89. package/template/packages/admin/src/components/ui/sidebar.tsx +740 -0
  90. package/template/packages/admin/src/components/ui/skeleton.tsx +13 -0
  91. package/template/packages/admin/src/components/ui/slider.tsx +63 -0
  92. package/template/packages/admin/src/components/ui/sonner.tsx +23 -0
  93. package/template/packages/admin/src/components/ui/switch.tsx +31 -0
  94. package/template/packages/admin/src/components/ui/table.tsx +114 -0
  95. package/template/packages/admin/src/components/ui/tabs.tsx +66 -0
  96. package/template/packages/admin/src/components/ui/textarea.tsx +18 -0
  97. package/template/packages/admin/src/components/ui/toggle-group.tsx +73 -0
  98. package/template/packages/admin/src/components/ui/toggle.tsx +45 -0
  99. package/template/packages/admin/src/components/ui/tooltip.tsx +59 -0
  100. package/template/packages/admin/src/hooks/use-mobile.ts +21 -0
  101. package/template/packages/admin/src/i18n.ts +20 -0
  102. package/template/packages/admin/src/index.tsx +18 -0
  103. package/template/packages/admin/src/layouts/menu-layout.tsx +211 -0
  104. package/template/packages/admin/src/layouts/tile-layout.tsx +355 -0
  105. package/template/packages/admin/src/lib/utils.ts +6 -0
  106. package/template/packages/admin/src/locales/en.json +68 -0
  107. package/template/packages/admin/src/locales/zh.json +68 -0
  108. package/template/packages/admin/src/pages/dashboard.tsx +12 -0
  109. package/template/packages/admin/src/pages/goods-receipt/CreatePage.tsx +302 -0
  110. package/template/packages/admin/src/pages/goods-receipt/EditPage.tsx +221 -0
  111. package/template/packages/admin/src/pages/goods-receipt/ListPage.tsx +283 -0
  112. package/template/packages/admin/src/pages/goods-receipt/ViewPage.tsx +280 -0
  113. package/template/packages/admin/src/pages/goods-receipt/index.ts +4 -0
  114. package/template/packages/admin/src/pages/home-page.tsx +244 -0
  115. package/template/packages/admin/src/pages/login-page.tsx +91 -0
  116. package/template/packages/admin/src/pages/master-data/cost-centers/index.tsx +461 -0
  117. package/template/packages/admin/src/pages/master-data/currencies/index.tsx +255 -0
  118. package/template/packages/admin/src/pages/master-data/materials/ListPage.tsx +271 -0
  119. package/template/packages/admin/src/pages/master-data/materials/ViewPage.tsx +240 -0
  120. package/template/packages/admin/src/pages/master-data/materials/index.ts +2 -0
  121. package/template/packages/admin/src/pages/master-data/plants/ListPage.tsx +279 -0
  122. package/template/packages/admin/src/pages/master-data/plants/ViewPage.tsx +380 -0
  123. package/template/packages/admin/src/pages/master-data/plants/index.ts +2 -0
  124. package/template/packages/admin/src/pages/master-data/purchase-organizations/index.tsx +341 -0
  125. package/template/packages/admin/src/pages/master-data/units-of-measure/index.tsx +295 -0
  126. package/template/packages/admin/src/pages/master-data/vendors/ListPage.tsx +266 -0
  127. package/template/packages/admin/src/pages/master-data/vendors/ViewPage.tsx +274 -0
  128. package/template/packages/admin/src/pages/master-data/vendors/index.ts +2 -0
  129. package/template/packages/admin/src/pages/placeholder-page.tsx +13 -0
  130. package/template/packages/admin/src/pages/purchase-orders/ListPage.tsx +289 -0
  131. package/template/packages/admin/src/pages/purchase-orders/ViewPage.tsx +343 -0
  132. package/template/packages/admin/src/pages/purchase-orders/index.ts +2 -0
  133. package/template/packages/admin/src/pages/purchase-requisitions/CreatePage.tsx +398 -0
  134. package/template/packages/admin/src/pages/purchase-requisitions/EditPage.tsx +473 -0
  135. package/template/packages/admin/src/pages/purchase-requisitions/ListPage.tsx +307 -0
  136. package/template/packages/admin/src/pages/purchase-requisitions/ViewPage.tsx +304 -0
  137. package/template/packages/admin/src/pages/purchase-requisitions/constants.ts +51 -0
  138. package/template/packages/admin/src/pages/purchase-requisitions/index.ts +4 -0
  139. package/template/packages/admin/src/pages/reports/PurchaseOrderReport.tsx +312 -0
  140. package/template/packages/admin/src/pages/reports/PurchaseRequisitionReport.tsx +303 -0
  141. package/template/packages/admin/src/pages/reports/index.ts +2 -0
  142. package/template/packages/admin/src/pages/settings-page.tsx +335 -0
  143. package/template/packages/admin/src/providers/app-config.tsx +50 -0
  144. package/template/packages/admin/src/routes/auth.ts +6 -0
  145. package/template/packages/admin/src/routes/index.ts +85 -0
  146. package/template/packages/admin/src/routes/menu.ts +176 -0
  147. package/template/packages/admin/src/routes/modules/goods-receipt.ts +31 -0
  148. package/template/packages/admin/src/routes/modules/master-data.ts +41 -0
  149. package/template/packages/admin/src/routes/modules/purchase-orders.ts +27 -0
  150. package/template/packages/admin/src/routes/modules/purchase-requisitions.ts +39 -0
  151. package/template/packages/admin/src/routes/modules/reports.ts +33 -0
  152. package/template/packages/admin/src/routes/modules/settings.ts +19 -0
  153. package/template/packages/admin/src/routes/withSuspense.tsx +21 -0
  154. package/template/packages/admin/src/theme/amber.css +27 -0
  155. package/template/packages/admin/src/theme/blue.css +27 -0
  156. package/template/packages/admin/src/theme/default.css +75 -0
  157. package/template/packages/admin/src/theme/fiori.css +180 -0
  158. package/template/packages/admin/src/theme/green.css +27 -0
  159. package/template/packages/admin/src/theme/index.css +12 -0
  160. package/template/packages/admin/src/theme/rose.css +27 -0
  161. package/template/packages/admin/src/theme/violet.css +27 -0
  162. package/template/packages/admin/src/vite-env.d.ts +1 -0
  163. package/template/packages/admin/tsconfig.json +28 -0
  164. package/template/packages/admin/tsconfig.node.json +21 -0
  165. package/template/packages/admin/vite.config.ts +26 -0
  166. package/template/packages/api/.eslintrc.json +6 -0
  167. package/template/packages/api/.swcrc +17 -0
  168. package/template/packages/api/app.config.ts +160 -0
  169. package/template/packages/api/docs/api-document.md +497 -0
  170. package/template/packages/api/examples/security/README.md +664 -0
  171. package/template/packages/api/examples/security/complete/.env.example +26 -0
  172. package/template/packages/api/examples/security/complete/PROJECT_STRUCTURE.md +220 -0
  173. package/template/packages/api/examples/security/complete/README.md +847 -0
  174. package/template/packages/api/examples/security/complete/app.config.ts +69 -0
  175. package/template/packages/api/examples/security/complete/app.ts +63 -0
  176. package/template/packages/api/examples/security/complete/controller/auth.controller.ts +131 -0
  177. package/template/packages/api/examples/security/complete/controller/index.ts +4 -0
  178. package/template/packages/api/examples/security/complete/controller/permission.controller.ts +41 -0
  179. package/template/packages/api/examples/security/complete/controller/role.controller.ts +53 -0
  180. package/template/packages/api/examples/security/complete/controller/user.controller.ts +53 -0
  181. package/template/packages/api/examples/security/complete/dto/change-password.dto.ts +10 -0
  182. package/template/packages/api/examples/security/complete/dto/create-permission.dto.ts +14 -0
  183. package/template/packages/api/examples/security/complete/dto/create-role.dto.ts +11 -0
  184. package/template/packages/api/examples/security/complete/dto/create-user.dto.ts +15 -0
  185. package/template/packages/api/examples/security/complete/dto/index.ts +7 -0
  186. package/template/packages/api/examples/security/complete/dto/login.dto.ts +10 -0
  187. package/template/packages/api/examples/security/complete/dto/oauth-profile.dto.ts +7 -0
  188. package/template/packages/api/examples/security/complete/dto/register.dto.ts +17 -0
  189. package/template/packages/api/examples/security/complete/entity/index.ts +6 -0
  190. package/template/packages/api/examples/security/complete/entity/oauth-account.entity.ts +39 -0
  191. package/template/packages/api/examples/security/complete/entity/permission.entity.ts +31 -0
  192. package/template/packages/api/examples/security/complete/entity/role-permission.entity.ts +19 -0
  193. package/template/packages/api/examples/security/complete/entity/role.entity.ts +25 -0
  194. package/template/packages/api/examples/security/complete/entity/user-role.entity.ts +19 -0
  195. package/template/packages/api/examples/security/complete/entity/user.entity.ts +46 -0
  196. package/template/packages/api/examples/security/complete/init.sql +81 -0
  197. package/template/packages/api/examples/security/complete/middleware/auth.interceptor.ts +39 -0
  198. package/template/packages/api/examples/security/complete/middleware/index.ts +2 -0
  199. package/template/packages/api/examples/security/complete/middleware/permission.interceptor.ts +61 -0
  200. package/template/packages/api/examples/security/complete/package.json +54 -0
  201. package/template/packages/api/examples/security/complete/seed.sql +42 -0
  202. package/template/packages/api/examples/security/complete/service/auth.service.ts +41 -0
  203. package/template/packages/api/examples/security/complete/service/index.ts +5 -0
  204. package/template/packages/api/examples/security/complete/service/oauth.service.ts +82 -0
  205. package/template/packages/api/examples/security/complete/service/permission.service.ts +113 -0
  206. package/template/packages/api/examples/security/complete/service/role.service.ts +85 -0
  207. package/template/packages/api/examples/security/complete/service/user.service.ts +132 -0
  208. package/template/packages/api/examples/security/complete/tests/TEST_REPORT.md +318 -0
  209. package/template/packages/api/examples/security/complete/tests/generate-report.js +335 -0
  210. package/template/packages/api/examples/security/complete/tests/helpers/api-helpers.ts +116 -0
  211. package/template/packages/api/examples/security/complete/tests/helpers/index.ts +2 -0
  212. package/template/packages/api/examples/security/complete/tests/helpers/test-helpers.ts +129 -0
  213. package/template/packages/api/examples/security/complete/tests/integration/auth.api.test.ts +429 -0
  214. package/template/packages/api/examples/security/complete/tests/integration/role.api.test.ts +400 -0
  215. package/template/packages/api/examples/security/complete/tests/integration/user.api.test.ts +459 -0
  216. package/template/packages/api/examples/security/complete/tests/jest.config.js +40 -0
  217. package/template/packages/api/examples/security/complete/tests/run-all-tests.js +135 -0
  218. package/template/packages/api/examples/security/complete/tests/run-tests.js +109 -0
  219. package/template/packages/api/examples/security/complete/tests/setup.ts +19 -0
  220. package/template/packages/api/examples/security/complete/tests/unit/auth.service.test.ts +199 -0
  221. package/template/packages/api/examples/security/complete/tests/unit/permission.service.test.ts +377 -0
  222. package/template/packages/api/examples/security/complete/tests/unit/user.service.test.ts +288 -0
  223. package/template/packages/api/examples/security/complete/tsconfig.json +35 -0
  224. package/template/packages/api/examples/security/jwt/README.md +424 -0
  225. package/template/packages/api/examples/security/local/README.md +499 -0
  226. package/template/packages/api/examples/security/oauth2/README.md +637 -0
  227. package/template/packages/api/examples/security/permission/README.md +943 -0
  228. package/template/packages/api/examples/security/session/README.md +753 -0
  229. package/template/packages/api/package.json +56 -0
  230. package/template/packages/api/src/controller/auth.controller.ts +127 -0
  231. package/template/packages/api/src/controller/cache.controller.ts +106 -0
  232. package/template/packages/api/src/controller/menu.controller.ts +46 -0
  233. package/template/packages/api/src/controller/mq.controller.ts +35 -0
  234. package/template/packages/api/src/controller/role.controller.ts +51 -0
  235. package/template/packages/api/src/controller/upload.controller.ts +85 -0
  236. package/template/packages/api/src/controller/user.controller.ts +57 -0
  237. package/template/packages/api/src/dto/auth.dto.ts +30 -0
  238. package/template/packages/api/src/dto/cache.dto.ts +24 -0
  239. package/template/packages/api/src/dto/menu.dto.ts +37 -0
  240. package/template/packages/api/src/dto/mq.dto.ts +16 -0
  241. package/template/packages/api/src/dto/role.dto.ts +16 -0
  242. package/template/packages/api/src/dto/user.dto.ts +35 -0
  243. package/template/packages/api/src/entity/menu.entity.ts +34 -0
  244. package/template/packages/api/src/entity/role-menu.entity.ts +13 -0
  245. package/template/packages/api/src/entity/role.entity.ts +22 -0
  246. package/template/packages/api/src/entity/user-role.entity.ts +13 -0
  247. package/template/packages/api/src/entity/user.entity.ts +31 -0
  248. package/template/packages/api/src/mapper/menu.mapper.ts +6 -0
  249. package/template/packages/api/src/mapper/role-menu.mapper.ts +6 -0
  250. package/template/packages/api/src/mapper/role.mapper.ts +6 -0
  251. package/template/packages/api/src/mapper/user-role.mapper.ts +6 -0
  252. package/template/packages/api/src/mapper/user.mapper.ts +11 -0
  253. package/template/packages/api/src/scripts/init-db.ts +185 -0
  254. package/template/packages/api/src/server.ts +76 -0
  255. package/template/packages/api/src/service/auth.service.ts +106 -0
  256. package/template/packages/api/src/service/cache.service.ts +80 -0
  257. package/template/packages/api/src/service/log.request.service.ts +158 -0
  258. package/template/packages/api/src/service/log.service.ts +123 -0
  259. package/template/packages/api/src/service/menu.service.ts +94 -0
  260. package/template/packages/api/src/service/mq.consumer.service.ts +26 -0
  261. package/template/packages/api/src/service/role.service.ts +88 -0
  262. package/template/packages/api/src/service/user.service.ts +170 -0
  263. package/template/packages/api/src/types/sqljs.d.ts +18 -0
  264. package/template/packages/api/src/utils/auth.utils.js +0 -0
  265. package/template/packages/api/src/utils/jwt.util.ts +29 -0
  266. package/template/packages/api/tsconfig.json +17 -0
  267. package/template/packages/api/tsup.config.ts +9 -0
  268. package/template/packages/api/uploads/.gitkeep +0 -0
  269. package/template/packages/core/package.json +31 -0
  270. package/template/packages/core/src/auth/auth-client-middleware.ts +22 -0
  271. package/template/packages/core/src/auth/auth-service.ts +65 -0
  272. package/template/packages/core/src/auth/default-auth-provider.ts +37 -0
  273. package/template/packages/core/src/auth/index.ts +10 -0
  274. package/template/packages/core/src/auth/types.ts +38 -0
  275. package/template/packages/core/src/authorization/authorization-client-middleware.ts +38 -0
  276. package/template/packages/core/src/authorization/authorization-config.ts +13 -0
  277. package/template/packages/core/src/authorization/authorization-provider.tsx +116 -0
  278. package/template/packages/core/src/authorization/default-authorization-provider.ts +26 -0
  279. package/template/packages/core/src/authorization/index.ts +15 -0
  280. package/template/packages/core/src/authorization/types.ts +42 -0
  281. package/template/packages/core/src/index.ts +3 -0
  282. package/template/packages/core/src/utils/promise-result-cache.ts +18 -0
  283. package/template/packages/core/tsconfig.json +19 -0
  284. package/template/packages/mobile/README.md +55 -0
  285. package/template/packages/mobile/index.html +13 -0
  286. package/template/packages/mobile/package.json +30 -0
  287. package/template/packages/mobile/postcss.config.mjs +7 -0
  288. package/template/packages/mobile/src/App.tsx +5 -0
  289. package/template/packages/mobile/src/app/globals.css +1 -0
  290. package/template/packages/mobile/src/components/LoginForm.tsx +76 -0
  291. package/template/packages/mobile/src/hooks/index.ts +5 -0
  292. package/template/packages/mobile/src/lib/utils.ts +7 -0
  293. package/template/packages/mobile/src/main.tsx +13 -0
  294. package/template/packages/mobile/src/pages/HomePage.tsx +35 -0
  295. package/template/packages/mobile/src/pages/LoginPage.tsx +35 -0
  296. package/template/packages/mobile/src/pages/index.ts +2 -0
  297. package/template/packages/mobile/src/routes/ProtectedRoute.tsx +29 -0
  298. package/template/packages/mobile/src/routes/index.tsx +24 -0
  299. package/template/packages/mobile/src/routes/routes.ts +11 -0
  300. package/template/packages/mobile/src/types/index.ts +5 -0
  301. package/template/packages/mobile/src/vite-env.d.ts +1 -0
  302. package/template/packages/mobile/tsconfig.json +23 -0
  303. package/template/packages/mobile/tsconfig.node.json +11 -0
  304. package/template/packages/mobile/vite.config.ts +19 -0
  305. package/template/packages/shared/package.json +20 -0
  306. package/template/packages/shared/src/constants.ts +8 -0
  307. package/template/packages/shared/src/index.ts +5 -0
  308. package/template/packages/shared/tsconfig.json +13 -0
  309. package/template/packages/shared-auth/package.json +29 -0
  310. package/template/packages/shared-auth/src/AuthContext.tsx +177 -0
  311. package/template/packages/shared-auth/src/AuthProviderWrapper.tsx +29 -0
  312. package/template/packages/shared-auth/src/index.ts +10 -0
  313. package/template/packages/shared-auth/tsconfig.json +14 -0
  314. package/template/pnpm-lock.yaml +16349 -0
  315. package/template/pnpm-workspace.yaml +3 -0
  316. package/template/scripts/postinstall.cjs +42 -0
  317. package/template/scripts/rebuild-sqlite.cjs +23 -0
@@ -0,0 +1,296 @@
1
+ "use client";
2
+
3
+ import * as React from "react";
4
+ import {
5
+ flexRender,
6
+ getCoreRowModel,
7
+ getPaginationRowModel,
8
+ getSortedRowModel,
9
+ useReactTable,
10
+ type ColumnDef,
11
+ type Row,
12
+ type SortingState,
13
+ } from "@tanstack/react-table";
14
+ import { cn } from "@/lib/utils";
15
+ import { Checkbox } from "@/components/ui/checkbox";
16
+ import {
17
+ Table,
18
+ TableBody,
19
+ TableCell,
20
+ TableHead,
21
+ TableHeader,
22
+ TableRow,
23
+ } from "@/components/ui/table";
24
+ import { DataTablePagination } from "@/components/admin-ui/data-table/data-table-pagination";
25
+ import { DataTableSorter } from "@/components/admin-ui/data-table/data-table-sorter";
26
+
27
+ export interface DataTableColumn<T> {
28
+ id: string;
29
+ header: string;
30
+ cell: (row: T) => React.ReactNode;
31
+ align?: "left" | "center" | "right";
32
+ /** 是否可排序,默认 true */
33
+ sortable?: boolean;
34
+ /** 排序依据的字段,不传则用 id 从 row 上取值 */
35
+ accessorKey?: keyof T;
36
+ }
37
+
38
+ export interface DataTableProps<T> {
39
+ data: T[];
40
+ columns: DataTableColumn<T>[];
41
+ getRowId?: (row: T) => string;
42
+ /** 是否显示分页,默认 true */
43
+ enablePagination?: boolean;
44
+ /** 服务端分页时启用,数据将按传入页直接渲染 */
45
+ manualPagination?: boolean;
46
+ pageSize?: number;
47
+ pageIndex?: number;
48
+ onPaginationChange?: (pageIndex: number, pageSize: number) => void;
49
+ /** 是否显示选择列 */
50
+ enableSelection?: boolean;
51
+ selectedIds?: Set<string>;
52
+ onSelectionChange?: (rows: T[]) => void;
53
+ loading?: boolean;
54
+ onRowClick?: (row: T) => void;
55
+ totalCount?: number;
56
+ className?: string;
57
+ }
58
+
59
+ function defaultGetRowId<T>(row: T): string {
60
+ return String((row as { id?: string }).id ?? "");
61
+ }
62
+
63
+ export function DataTable<T extends object>({
64
+ data,
65
+ columns,
66
+ getRowId = defaultGetRowId,
67
+ enablePagination = true,
68
+ manualPagination = false,
69
+ pageSize: initialPageSize = 10,
70
+ pageIndex: controlledPageIndex,
71
+ onPaginationChange,
72
+ enableSelection = false,
73
+ selectedIds = new Set(),
74
+ onSelectionChange,
75
+ loading = false,
76
+ onRowClick,
77
+ totalCount,
78
+ className,
79
+ }: DataTableProps<T>) {
80
+ const [sorting, setSorting] = React.useState<SortingState>([]);
81
+ const [pagination, setPagination] = React.useState({
82
+ pageIndex: controlledPageIndex ?? 0,
83
+ pageSize: initialPageSize,
84
+ });
85
+
86
+ React.useEffect(() => {
87
+ setPagination((prev) => {
88
+ const nextPageIndex = controlledPageIndex ?? 0;
89
+ if (prev.pageIndex === nextPageIndex && prev.pageSize === initialPageSize) {
90
+ return prev;
91
+ }
92
+
93
+ return {
94
+ pageIndex: nextPageIndex,
95
+ pageSize: initialPageSize,
96
+ };
97
+ });
98
+ }, [controlledPageIndex, initialPageSize]);
99
+
100
+ const columnDefs = React.useMemo<ColumnDef<T>[]>(() => {
101
+ const cols: ColumnDef<T>[] = [];
102
+ if (enableSelection) {
103
+ cols.push({
104
+ id: "select",
105
+ size: 40,
106
+ accessorFn: () => undefined,
107
+ header: ({ table }) => (
108
+ <Checkbox
109
+ checked={
110
+ table.getIsAllPageRowsSelected() ||
111
+ (table.getIsSomePageRowsSelected() && "indeterminate")
112
+ }
113
+ onCheckedChange={(v) => table.toggleAllPageRowsSelected(!!v)}
114
+ aria-label="全选"
115
+ />
116
+ ),
117
+ cell: ({ row }) => (
118
+ <Checkbox
119
+ checked={row.getIsSelected()}
120
+ onCheckedChange={(v) => row.toggleSelected(!!v)}
121
+ aria-label="选择行"
122
+ onClick={(e) => e.stopPropagation()}
123
+ />
124
+ ),
125
+ enableSorting: false,
126
+ });
127
+ }
128
+ columns.forEach((col) => {
129
+ const key = col.accessorKey ?? (col.id as keyof T);
130
+ const keyStr = String(key);
131
+ cols.push({
132
+ id: col.id,
133
+ accessorKey: keyStr,
134
+ accessorFn: (row: T) =>
135
+ (row as Record<string, unknown>)[keyStr] as unknown,
136
+ header: col.header,
137
+ cell: ({ row }) => col.cell(row.original),
138
+ enableSorting: col.sortable === true,
139
+ meta: { align: col.align },
140
+ });
141
+ });
142
+ return cols;
143
+ }, [columns, enableSelection]);
144
+
145
+ const rowSelection = React.useMemo(
146
+ () =>
147
+ enableSelection && selectedIds
148
+ ? Object.fromEntries([...selectedIds].map((id) => [id, true]))
149
+ : {},
150
+ [enableSelection, selectedIds]
151
+ );
152
+
153
+ const manualPageCount = React.useMemo(() => {
154
+ if (!manualPagination) {
155
+ return undefined;
156
+ }
157
+
158
+ if (!totalCount || pagination.pageSize <= 0) {
159
+ return 0;
160
+ }
161
+
162
+ return Math.ceil(totalCount / pagination.pageSize);
163
+ }, [manualPagination, pagination.pageSize, totalCount]);
164
+
165
+ const table = useReactTable({
166
+ data,
167
+ columns: columnDefs,
168
+ getRowId: (row: T) => getRowId(row),
169
+ state: {
170
+ sorting,
171
+ pagination,
172
+ ...(enableSelection ? { rowSelection } : {}),
173
+ },
174
+ onSortingChange: setSorting,
175
+ onPaginationChange: (updater) => {
176
+ const next =
177
+ typeof updater === "function" ? updater(pagination) : updater;
178
+ setPagination(next);
179
+ onPaginationChange?.(next.pageIndex, next.pageSize);
180
+ },
181
+ onRowSelectionChange: enableSelection
182
+ ? (updater) => {
183
+ const next = typeof updater === "function" ? updater(rowSelection) : updater;
184
+ const ids = Object.keys(next).filter((id) => next[id]);
185
+ const rows = data.filter((r) => ids.includes(getRowId(r)));
186
+ onSelectionChange?.(rows);
187
+ }
188
+ : undefined,
189
+ getCoreRowModel: getCoreRowModel(),
190
+ getSortedRowModel: getSortedRowModel(),
191
+ getPaginationRowModel: enablePagination && !manualPagination ? getPaginationRowModel() : undefined,
192
+ manualPagination,
193
+ pageCount: manualPageCount,
194
+ });
195
+
196
+ const rows = table.getRowModel().rows;
197
+ const pageCount = manualPagination ? (manualPageCount ?? 0) : table.getPageCount();
198
+ const currentPage = table.getState().pagination.pageIndex + 1;
199
+
200
+ return (
201
+ <div className={cn("overflow-hidden", className)}>
202
+ <Table>
203
+ <TableHeader>
204
+ {table.getHeaderGroups().map((headerGroup) => (
205
+ <TableRow key={headerGroup.id}>
206
+ {headerGroup.headers.map((header) => (
207
+ <TableHead
208
+ key={header.id}
209
+ className={cn(
210
+ header.id === "select" && "w-10",
211
+ (header.column.columnDef.meta as { align?: string })?.align ===
212
+ "center" && "text-center",
213
+ (header.column.columnDef.meta as { align?: string })?.align ===
214
+ "right" && "text-right"
215
+ )}
216
+ >
217
+ <div className="flex items-center gap-1">
218
+ {flexRender(
219
+ header.column.columnDef.header,
220
+ header.getContext()
221
+ )}
222
+ {header.column.getCanSort() && (
223
+ <DataTableSorter column={header.column} />
224
+ )}
225
+ </div>
226
+ </TableHead>
227
+ ))}
228
+ </TableRow>
229
+ ))}
230
+ </TableHeader>
231
+ <TableBody>
232
+ {loading ? (
233
+ <TableRow>
234
+ <TableCell
235
+ colSpan={columnDefs.length}
236
+ className="h-24 text-center"
237
+ >
238
+ 加载中...
239
+ </TableCell>
240
+ </TableRow>
241
+ ) : rows.length === 0 ? (
242
+ <TableRow>
243
+ <TableCell
244
+ colSpan={columnDefs.length}
245
+ className="h-24 text-center text-muted-foreground"
246
+ >
247
+ 暂无数据
248
+ </TableCell>
249
+ </TableRow>
250
+ ) : (
251
+ rows.map((row) => (
252
+ <TableRow
253
+ key={row.id}
254
+ data-state={row.getIsSelected() ? "selected" : undefined}
255
+ className={onRowClick ? "cursor-pointer" : undefined}
256
+ onClick={() => onRowClick?.(row.original as T)}
257
+ >
258
+ {row.getVisibleCells().map((cell) => (
259
+ <TableCell
260
+ key={cell.id}
261
+ className={cn(
262
+ (cell.column.columnDef.meta as { align?: string })?.align ===
263
+ "center" && "text-center",
264
+ (cell.column.columnDef.meta as { align?: string })?.align ===
265
+ "right" && "text-right"
266
+ )}
267
+ >
268
+ {flexRender(
269
+ cell.column.columnDef.cell,
270
+ cell.getContext()
271
+ )}
272
+ </TableCell>
273
+ ))}
274
+ </TableRow>
275
+ ))
276
+ )}
277
+ </TableBody>
278
+ </Table>
279
+
280
+ {enablePagination && pageCount > 0 && (
281
+ <div className="p-3">
282
+ <DataTablePagination
283
+ currentPage={currentPage}
284
+ pageCount={pageCount}
285
+ setCurrentPage={(p) => table.setPageIndex(p - 1)}
286
+ pageSize={table.getState().pagination.pageSize}
287
+ setPageSize={(size) => table.setPageSize(size)}
288
+ total={totalCount ?? data.length}
289
+ />
290
+ </div>
291
+ )}
292
+ </div>
293
+ );
294
+ }
295
+
296
+ DataTable.displayName = "DataTable";
@@ -0,0 +1,292 @@
1
+ "use client";
2
+
3
+ import type { ReactNode } from "react";
4
+ import { cn } from "@/lib/utils";
5
+ import {
6
+ Select,
7
+ SelectContent,
8
+ SelectItem,
9
+ SelectTrigger,
10
+ SelectValue,
11
+ } from "@/components/ui/select";
12
+ import {
13
+ Table,
14
+ TableBody,
15
+ TableCell,
16
+ TableHead,
17
+ TableHeader,
18
+ TableRow,
19
+ } from "@/components/ui/table";
20
+ import { Input } from "@/components/ui/input";
21
+ import { Button } from "@/components/ui/button";
22
+ import { Trash2 } from "lucide-react";
23
+
24
+ export interface EditableTableColumn<T> {
25
+ key: string;
26
+ title: string | ReactNode;
27
+ width?: string | number;
28
+ align?: "left" | "center" | "right";
29
+ required?: boolean;
30
+ render: (record: T, index: number) => ReactNode;
31
+ }
32
+
33
+ export interface EditableTableProps<T> {
34
+ columns: EditableTableColumn<T>[];
35
+ dataSource: T[];
36
+ rowKey: keyof T | ((record: T) => string);
37
+ header?: { title: string; subtitle?: string; actions?: ReactNode };
38
+ footer?: ReactNode;
39
+ emptyText?: string;
40
+ minWidth?: number;
41
+ className?: string;
42
+ showIndex?: boolean;
43
+ embedded?: boolean;
44
+ }
45
+
46
+ export function EditableTable<T>({
47
+ columns,
48
+ dataSource,
49
+ rowKey,
50
+ header,
51
+ footer,
52
+ emptyText = "暂无数据",
53
+ minWidth = 900,
54
+ className,
55
+ showIndex = true,
56
+ embedded = false,
57
+ }: EditableTableProps<T>) {
58
+ const getRowKey = (record: T, index: number): string => {
59
+ if (typeof rowKey === "function") return rowKey(record);
60
+ return String(record[rowKey] ?? index);
61
+ };
62
+
63
+ return (
64
+ <div
65
+ className={cn(
66
+ "overflow-hidden",
67
+ !embedded && "rounded-xl border border-border bg-card shadow-sm",
68
+ embedded && "-mx-6 -mb-6 -mt-6",
69
+ className
70
+ )}
71
+ >
72
+ {header && (
73
+ <div className="flex items-center justify-between border-b border-border bg-muted/30 px-6 py-4">
74
+ <div>
75
+ <h2 className="font-semibold">{header.title}</h2>
76
+ {header.subtitle && (
77
+ <p className="mt-0.5 text-xs text-muted-foreground">{header.subtitle}</p>
78
+ )}
79
+ </div>
80
+ {header.actions && <div className="flex items-center gap-2">{header.actions}</div>}
81
+ </div>
82
+ )}
83
+ <div className="overflow-x-auto">
84
+ <Table style={{ minWidth }}>
85
+ <TableHeader>
86
+ <TableRow>
87
+ {showIndex && (
88
+ <TableHead className="w-10 px-3 py-3 text-left text-xs font-semibold">#</TableHead>
89
+ )}
90
+ {columns.map((col) => (
91
+ <TableHead
92
+ key={col.key}
93
+ className={cn(
94
+ "px-3 py-3 text-xs font-semibold",
95
+ col.align === "center" && "text-center",
96
+ col.align === "right" && "text-right"
97
+ )}
98
+ style={{ width: col.width }}
99
+ >
100
+ {col.title}
101
+ {col.required && <span className="ml-0.5 text-destructive">*</span>}
102
+ </TableHead>
103
+ ))}
104
+ </TableRow>
105
+ </TableHeader>
106
+ <TableBody>
107
+ {dataSource.length > 0 ? (
108
+ dataSource.map((record, index) => (
109
+ <TableRow key={getRowKey(record, index)}>
110
+ {showIndex && (
111
+ <TableCell className="px-3 py-2 text-sm text-muted-foreground">
112
+ {index + 1}
113
+ </TableCell>
114
+ )}
115
+ {columns.map((col) => (
116
+ <TableCell
117
+ key={col.key}
118
+ className={cn(
119
+ "px-3 py-2",
120
+ col.align === "center" && "text-center",
121
+ col.align === "right" && "text-right"
122
+ )}
123
+ >
124
+ {col.render(record, index)}
125
+ </TableCell>
126
+ ))}
127
+ </TableRow>
128
+ ))
129
+ ) : (
130
+ <TableRow>
131
+ <TableCell
132
+ colSpan={columns.length + (showIndex ? 1 : 0)}
133
+ className="px-3 py-12 text-center text-muted-foreground"
134
+ >
135
+ {emptyText}
136
+ </TableCell>
137
+ </TableRow>
138
+ )}
139
+ </TableBody>
140
+ {footer && dataSource.length > 0 && <tfoot>{footer}</tfoot>}
141
+ </Table>
142
+ </div>
143
+ </div>
144
+ );
145
+ }
146
+
147
+ export function TableInput({
148
+ type = "text",
149
+ value,
150
+ onChange,
151
+ readOnly,
152
+ placeholder,
153
+ className,
154
+ align = "left",
155
+ ...props
156
+ }: {
157
+ type?: "text" | "number" | "date";
158
+ value: string | number;
159
+ onChange?: (value: string) => void;
160
+ readOnly?: boolean;
161
+ placeholder?: string;
162
+ className?: string;
163
+ align?: "left" | "center" | "right";
164
+ min?: number;
165
+ max?: number;
166
+ }) {
167
+ return (
168
+ <Input
169
+ type={type}
170
+ value={value}
171
+ onChange={(e) => onChange?.(e.target.value)}
172
+ readOnly={readOnly}
173
+ placeholder={placeholder}
174
+ className={cn(
175
+ "h-9",
176
+ readOnly && "bg-muted text-muted-foreground",
177
+ align === "center" && "text-center",
178
+ align === "right" && "text-right",
179
+ className
180
+ )}
181
+ {...props}
182
+ />
183
+ );
184
+ }
185
+
186
+ /** Sentinel for placeholder option; Radix Select forbids SelectItem value="". */
187
+ const PLACEHOLDER_VALUE = "__select_placeholder__";
188
+
189
+ export function TableSelect({
190
+ value,
191
+ onChange,
192
+ options,
193
+ placeholder,
194
+ className,
195
+ }: {
196
+ value: string;
197
+ onChange: (value: string) => void;
198
+ options: Array<{ value: string; label: string }>;
199
+ placeholder?: string;
200
+ className?: string;
201
+ }) {
202
+ const items = placeholder
203
+ ? [{ value: PLACEHOLDER_VALUE, label: placeholder }, ...options]
204
+ : options;
205
+ const selectValue = value || (placeholder ? PLACEHOLDER_VALUE : undefined);
206
+ return (
207
+ <Select
208
+ value={selectValue}
209
+ onValueChange={(v) => onChange(v === PLACEHOLDER_VALUE ? "" : v)}
210
+ >
211
+ <SelectTrigger className={cn("h-9", className)}>
212
+ <SelectValue placeholder={placeholder} />
213
+ </SelectTrigger>
214
+ <SelectContent>
215
+ {items.map((opt) => (
216
+ <SelectItem key={opt.value} value={opt.value}>
217
+ {opt.label}
218
+ </SelectItem>
219
+ ))}
220
+ </SelectContent>
221
+ </Select>
222
+ );
223
+ }
224
+
225
+ export function TableText({
226
+ children,
227
+ className,
228
+ variant = "default",
229
+ }: {
230
+ children: ReactNode;
231
+ className?: string;
232
+ variant?: "default" | "muted" | "mono" | "bold" | "primary";
233
+ }) {
234
+ const variantStyles = {
235
+ default: "text-sm",
236
+ muted: "text-sm text-muted-foreground",
237
+ mono: "font-mono text-xs text-muted-foreground",
238
+ bold: "text-sm font-medium",
239
+ primary: "text-sm font-semibold text-primary",
240
+ };
241
+ return <span className={cn(variantStyles[variant], className)}>{children}</span>;
242
+ }
243
+
244
+ export function TableDeleteButton({
245
+ onClick,
246
+ disabled,
247
+ }: {
248
+ onClick: () => void;
249
+ disabled?: boolean;
250
+ }) {
251
+ return (
252
+ <Button
253
+ type="button"
254
+ variant="ghost"
255
+ size="icon"
256
+ className={cn(
257
+ "size-8",
258
+ !disabled && "text-muted-foreground hover:text-destructive hover:bg-destructive/10"
259
+ )}
260
+ onClick={onClick}
261
+ disabled={disabled}
262
+ >
263
+ <Trash2 className="size-4" />
264
+ </Button>
265
+ );
266
+ }
267
+
268
+ export function TableFooterRow({
269
+ label,
270
+ value,
271
+ colSpan,
272
+ valueColSpan = 1,
273
+ tailColSpan = 0,
274
+ }: {
275
+ label: string;
276
+ value: ReactNode;
277
+ colSpan: number;
278
+ valueColSpan?: number;
279
+ tailColSpan?: number;
280
+ }) {
281
+ return (
282
+ <TableRow className="bg-muted/50 border-t border-border">
283
+ <TableCell colSpan={colSpan} className="px-3 py-3 text-right text-sm font-medium">
284
+ {label}
285
+ </TableCell>
286
+ <TableCell colSpan={valueColSpan} className="px-3 py-3 text-right">
287
+ {value}
288
+ </TableCell>
289
+ {tailColSpan > 0 && <TableCell colSpan={tailColSpan} />}
290
+ </TableRow>
291
+ );
292
+ }
@@ -0,0 +1,38 @@
1
+ "use client";
2
+
3
+ import { useState } from "react";
4
+ import { Eye, EyeOff } from "lucide-react";
5
+ import { Input } from "@/components/ui/input";
6
+ import { cn } from "@/lib/utils";
7
+
8
+ type InputPasswordProps = React.ComponentProps<"input">;
9
+
10
+ export const InputPassword = ({ className, ...props }: InputPasswordProps) => {
11
+ const [showPassword, setShowPassword] = useState(false);
12
+
13
+ return (
14
+ <div className={cn("relative")}>
15
+ <Input
16
+ type={showPassword ? "text" : "password"}
17
+ className={cn(className)}
18
+ {...props}
19
+ />
20
+ <button
21
+ type="button"
22
+ className={cn(
23
+ "appearance-none",
24
+ "absolute right-3 top-1/2 -translate-y-1/2"
25
+ )}
26
+ onClick={() => setShowPassword(!showPassword)}
27
+ >
28
+ {showPassword ? (
29
+ <EyeOff size={18} className={cn("text-gray-500")} />
30
+ ) : (
31
+ <Eye size={18} className={cn("text-gray-500")} />
32
+ )}
33
+ </button>
34
+ </div>
35
+ );
36
+ };
37
+
38
+ InputPassword.displayName = "InputPassword";