@vendure/dashboard 3.2.0 → 3.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (381) hide show
  1. package/LICENSE.md +42 -0
  2. package/README.md +12 -12
  3. package/index.html +15 -15
  4. package/lingui.config.js +12 -12
  5. package/package.json +11 -2
  6. package/src/app/app-providers.tsx +30 -30
  7. package/src/app/main.tsx +97 -97
  8. package/src/app/routes/__root.tsx +24 -24
  9. package/src/app/routes/_authenticated/_administrators/administrators.graphql.ts +79 -79
  10. package/src/app/routes/_authenticated/_administrators/administrators.tsx +86 -86
  11. package/src/app/routes/_authenticated/_administrators/administrators_.$id.tsx +155 -155
  12. package/src/app/routes/_authenticated/_administrators/components/role-permissions-display.tsx +133 -133
  13. package/src/app/routes/_authenticated/_assets/assets.tsx +19 -19
  14. package/src/app/routes/_authenticated/_channels/channels.graphql.ts +93 -93
  15. package/src/app/routes/_authenticated/_channels/channels.tsx +60 -60
  16. package/src/app/routes/_authenticated/_channels/channels_.$id.tsx +248 -248
  17. package/src/app/routes/_authenticated/_collections/collections.graphql.ts +133 -133
  18. package/src/app/routes/_authenticated/_collections/collections.tsx +195 -195
  19. package/src/app/routes/_authenticated/_collections/collections_.$id.tsx +224 -224
  20. package/src/app/routes/_authenticated/_collections/components/collection-contents-preview-table.tsx +127 -127
  21. package/src/app/routes/_authenticated/_collections/components/collection-contents-sheet.tsx +46 -46
  22. package/src/app/routes/_authenticated/_collections/components/collection-contents-table.tsx +82 -82
  23. package/src/app/routes/_authenticated/_collections/components/collection-filters-selector.tsx +91 -91
  24. package/src/app/routes/_authenticated/_countries/countries.graphql.ts +69 -69
  25. package/src/app/routes/_authenticated/_countries/countries.tsx +67 -67
  26. package/src/app/routes/_authenticated/_countries/countries_.$id.tsx +122 -122
  27. package/src/app/routes/_authenticated/_customer-groups/components/customer-group-members-sheet.tsx +44 -44
  28. package/src/app/routes/_authenticated/_customer-groups/components/customer-group-members-table.tsx +129 -129
  29. package/src/app/routes/_authenticated/_customer-groups/customer-groups.graphql.ts +71 -71
  30. package/src/app/routes/_authenticated/_customer-groups/customer-groups.tsx +68 -68
  31. package/src/app/routes/_authenticated/_customer-groups/customer-groups_.$id.tsx +111 -111
  32. package/src/app/routes/_authenticated/_customers/components/customer-address-card.tsx +155 -155
  33. package/src/app/routes/_authenticated/_customers/components/customer-address-form.tsx +344 -344
  34. package/src/app/routes/_authenticated/_customers/components/customer-group-controls.tsx +4 -4
  35. package/src/app/routes/_authenticated/_customers/components/customer-history/customer-history-container.tsx +78 -78
  36. package/src/app/routes/_authenticated/_customers/components/customer-history/customer-history.tsx +77 -77
  37. package/src/app/routes/_authenticated/_customers/components/customer-history/index.ts +3 -3
  38. package/src/app/routes/_authenticated/_customers/components/customer-history/use-customer-history.ts +169 -169
  39. package/src/app/routes/_authenticated/_customers/components/customer-order-table.tsx +88 -88
  40. package/src/app/routes/_authenticated/_customers/components/customer-status-badge.tsx +33 -33
  41. package/src/app/routes/_authenticated/_customers/customers.graphql.ts +204 -204
  42. package/src/app/routes/_authenticated/_customers/customers.tsx +82 -82
  43. package/src/app/routes/_authenticated/_customers/customers_.$id.tsx +274 -274
  44. package/src/app/routes/_authenticated/_facets/components/edit-facet-value.tsx +129 -129
  45. package/src/app/routes/_authenticated/_facets/components/facet-values-sheet.tsx +46 -46
  46. package/src/app/routes/_authenticated/_facets/components/facet-values-table.tsx +97 -97
  47. package/src/app/routes/_authenticated/_facets/facets.graphql.ts +104 -104
  48. package/src/app/routes/_authenticated/_facets/facets.tsx +97 -97
  49. package/src/app/routes/_authenticated/_facets/facets_.$id.tsx +139 -139
  50. package/src/app/routes/_authenticated/_global-settings/global-settings.graphql.ts +28 -28
  51. package/src/app/routes/_authenticated/_global-settings/global-settings.tsx +161 -161
  52. package/src/app/routes/_authenticated/_orders/components/order-address.tsx +58 -58
  53. package/src/app/routes/_authenticated/_orders/components/order-history/index.ts +3 -3
  54. package/src/app/routes/_authenticated/_orders/components/order-history/order-history-container.tsx +72 -72
  55. package/src/app/routes/_authenticated/_orders/components/order-history/order-history.tsx +96 -96
  56. package/src/app/routes/_authenticated/_orders/components/order-history/use-order-history.ts +171 -171
  57. package/src/app/routes/_authenticated/_orders/components/order-table.tsx +169 -169
  58. package/src/app/routes/_authenticated/_orders/components/order-tax-summary.tsx +38 -38
  59. package/src/app/routes/_authenticated/_orders/components/payment-details.tsx +61 -61
  60. package/src/app/routes/_authenticated/_orders/orders.graphql.ts +325 -325
  61. package/src/app/routes/_authenticated/_orders/orders.tsx +120 -120
  62. package/src/app/routes/_authenticated/_orders/orders_.$id.tsx +133 -133
  63. package/src/app/routes/_authenticated/_payment-methods/components/payment-eligibility-checker-selector.tsx +104 -104
  64. package/src/app/routes/_authenticated/_payment-methods/components/payment-handler-selector.tsx +100 -100
  65. package/src/app/routes/_authenticated/_payment-methods/payment-methods.graphql.ts +83 -83
  66. package/src/app/routes/_authenticated/_payment-methods/payment-methods.tsx +64 -64
  67. package/src/app/routes/_authenticated/_payment-methods/payment-methods_.$id.tsx +183 -183
  68. package/src/app/routes/_authenticated/_product-variants/components/variant-price-detail.tsx +87 -87
  69. package/src/app/routes/_authenticated/_product-variants/product-variants.graphql.ts +123 -123
  70. package/src/app/routes/_authenticated/_product-variants/product-variants.tsx +78 -78
  71. package/src/app/routes/_authenticated/_product-variants/product-variants_.$id.tsx +331 -331
  72. package/src/app/routes/_authenticated/_products/components/create-product-variants-dialog.tsx +228 -228
  73. package/src/app/routes/_authenticated/_products/components/create-product-variants.tsx +462 -462
  74. package/src/app/routes/_authenticated/_products/components/option-value-input.tsx +95 -95
  75. package/src/app/routes/_authenticated/_products/components/product-variants-table.tsx +87 -87
  76. package/src/app/routes/_authenticated/_products/products.graphql.ts +116 -116
  77. package/src/app/routes/_authenticated/_products/products.tsx +48 -48
  78. package/src/app/routes/_authenticated/_products/products_.$id.tsx +196 -196
  79. package/src/app/routes/_authenticated/_profile/profile.graphql.ts +23 -23
  80. package/src/app/routes/_authenticated/_profile/profile.tsx +122 -122
  81. package/src/app/routes/_authenticated/_promotions/components/promotion-actions-selector.tsx +107 -107
  82. package/src/app/routes/_authenticated/_promotions/components/promotion-conditions-selector.tsx +107 -107
  83. package/src/app/routes/_authenticated/_promotions/promotions.graphql.ts +96 -96
  84. package/src/app/routes/_authenticated/_promotions/promotions.tsx +61 -61
  85. package/src/app/routes/_authenticated/_promotions/promotions_.$id.tsx +235 -235
  86. package/src/app/routes/_authenticated/_roles/components/expandable-permissions.tsx +54 -54
  87. package/src/app/routes/_authenticated/_roles/components/permissions-grid.tsx +116 -116
  88. package/src/app/routes/_authenticated/_roles/roles.graphql.ts +67 -67
  89. package/src/app/routes/_authenticated/_roles/roles.tsx +96 -96
  90. package/src/app/routes/_authenticated/_roles/roles_.$id.tsx +142 -142
  91. package/src/app/routes/_authenticated/_sellers/sellers.graphql.ts +61 -61
  92. package/src/app/routes/_authenticated/_sellers/sellers.tsx +51 -51
  93. package/src/app/routes/_authenticated/_sellers/sellers_.$id.tsx +111 -111
  94. package/src/app/routes/_authenticated/_shipping-methods/components/fulfillment-handler-selector.tsx +56 -56
  95. package/src/app/routes/_authenticated/_shipping-methods/components/shipping-calculator-selector.tsx +101 -101
  96. package/src/app/routes/_authenticated/_shipping-methods/components/shipping-eligibility-checker-selector.tsx +101 -101
  97. package/src/app/routes/_authenticated/_shipping-methods/components/test-shipping-method-dialog.tsx +32 -32
  98. package/src/app/routes/_authenticated/_shipping-methods/shipping-methods.graphql.ts +83 -83
  99. package/src/app/routes/_authenticated/_shipping-methods/shipping-methods.tsx +55 -55
  100. package/src/app/routes/_authenticated/_shipping-methods/shipping-methods_.$id.tsx +171 -171
  101. package/src/app/routes/_authenticated/_stock-locations/stock-locations.graphql.ts +62 -62
  102. package/src/app/routes/_authenticated/_stock-locations/stock-locations.tsx +48 -48
  103. package/src/app/routes/_authenticated/_stock-locations/stock-locations_.$id.tsx +115 -115
  104. package/src/app/routes/_authenticated/_system/components/payload-dialog.tsx +34 -34
  105. package/src/app/routes/_authenticated/_system/healthchecks.tsx +93 -93
  106. package/src/app/routes/_authenticated/_system/job-queue.graphql.ts +43 -43
  107. package/src/app/routes/_authenticated/_system/job-queue.tsx +161 -161
  108. package/src/app/routes/_authenticated/_tax-categories/tax-categories.graphql.ts +63 -63
  109. package/src/app/routes/_authenticated/_tax-categories/tax-categories.tsx +65 -65
  110. package/src/app/routes/_authenticated/_tax-categories/tax-categories_.$id.tsx +115 -115
  111. package/src/app/routes/_authenticated/_tax-rates/tax-rates.graphql.ts +75 -75
  112. package/src/app/routes/_authenticated/_tax-rates/tax-rates.tsx +108 -108
  113. package/src/app/routes/_authenticated/_tax-rates/tax-rates_.$id.tsx +148 -148
  114. package/src/app/routes/_authenticated/_zones/components/zone-countries-sheet.tsx +31 -31
  115. package/src/app/routes/_authenticated/_zones/components/zone-countries-table.tsx +79 -79
  116. package/src/app/routes/_authenticated/_zones/zones.graphql.ts +96 -96
  117. package/src/app/routes/_authenticated/_zones/zones.tsx +57 -57
  118. package/src/app/routes/_authenticated/_zones/zones_.$id.tsx +103 -103
  119. package/src/app/routes/_authenticated/index.tsx +194 -194
  120. package/src/app/routes/_authenticated.tsx +25 -25
  121. package/src/app/routes/login.tsx +48 -48
  122. package/src/app/styles.css +82 -82
  123. package/src/app/tailwindcss-animate.css +275 -275
  124. package/src/i18n/locales/de.po +1579 -1579
  125. package/src/i18n/locales/en.po +1579 -1579
  126. package/src/lib/components/data-display/boolean.tsx +23 -23
  127. package/src/lib/components/data-display/date-time.tsx +13 -13
  128. package/src/lib/components/data-display/json.tsx +5 -5
  129. package/src/lib/components/data-display/money.tsx +15 -15
  130. package/src/lib/components/data-input/affixed-input.tsx +49 -49
  131. package/src/lib/components/data-input/customer-group-input.tsx +72 -72
  132. package/src/lib/components/data-input/datetime-input.tsx +149 -149
  133. package/src/lib/components/data-input/facet-value-input.tsx +68 -68
  134. package/src/lib/components/data-input/money-input.tsx +112 -112
  135. package/src/lib/components/data-input/richt-text-input.tsx +99 -99
  136. package/src/lib/components/data-table/data-table-column-header.tsx +73 -73
  137. package/src/lib/components/data-table/data-table-faceted-filter.tsx +172 -172
  138. package/src/lib/components/data-table/data-table-filter-dialog.tsx +73 -73
  139. package/src/lib/components/data-table/data-table-pagination.tsx +87 -87
  140. package/src/lib/components/data-table/data-table-view-options.tsx +53 -53
  141. package/src/lib/components/data-table/data-table.tsx +218 -218
  142. package/src/lib/components/layout/app-layout.tsx +42 -42
  143. package/src/lib/components/layout/app-sidebar.tsx +34 -34
  144. package/src/lib/components/layout/channel-switcher.tsx +87 -87
  145. package/src/lib/components/layout/content-language-selector.tsx +41 -41
  146. package/src/lib/components/layout/generated-breadcrumbs.tsx +82 -82
  147. package/src/lib/components/layout/language-dialog.tsx +104 -104
  148. package/src/lib/components/layout/nav-main.tsx +178 -178
  149. package/src/lib/components/layout/nav-projects.tsx +81 -81
  150. package/src/lib/components/layout/nav-user.tsx +176 -176
  151. package/src/lib/components/layout/prerelease-popup.tsx +38 -38
  152. package/src/lib/components/login/login-form.tsx +173 -173
  153. package/src/lib/components/shared/alerts.tsx +20 -20
  154. package/src/lib/components/shared/animated-number.tsx +49 -49
  155. package/src/lib/components/shared/asset-gallery.tsx +433 -433
  156. package/src/lib/components/shared/asset-picker-dialog.tsx +71 -71
  157. package/src/lib/components/shared/asset-preview-dialog.tsx +48 -48
  158. package/src/lib/components/shared/asset-preview.tsx +345 -345
  159. package/src/lib/components/shared/assigned-facet-values.tsx +68 -68
  160. package/src/lib/components/shared/channel-code-label.tsx +7 -7
  161. package/src/lib/components/shared/channel-selector.tsx +51 -51
  162. package/src/lib/components/shared/configurable-operation-arg-input.tsx +51 -51
  163. package/src/lib/components/shared/configurable-operation-input.tsx +133 -133
  164. package/src/lib/components/shared/confirmation-dialog.tsx +58 -58
  165. package/src/lib/components/shared/copyable-text.tsx +31 -31
  166. package/src/lib/components/shared/country-selector.tsx +105 -105
  167. package/src/lib/components/shared/currency-selector.tsx +33 -33
  168. package/src/lib/components/shared/custom-fields-form.tsx +86 -86
  169. package/src/lib/components/shared/customer-address-form.tsx +330 -330
  170. package/src/lib/components/shared/customer-group-chip.tsx +30 -30
  171. package/src/lib/components/shared/customer-group-selector.tsx +62 -62
  172. package/src/lib/components/shared/customer-selector.tsx +107 -107
  173. package/src/lib/components/shared/detail-page-button.tsx +22 -22
  174. package/src/lib/components/shared/entity-assets.tsx +340 -340
  175. package/src/lib/components/shared/error-page.tsx +31 -31
  176. package/src/lib/components/shared/facet-value-chip.tsx +44 -44
  177. package/src/lib/components/shared/facet-value-selector.tsx +306 -306
  178. package/src/lib/components/shared/focal-point-control.tsx +64 -64
  179. package/src/lib/components/shared/form-field-wrapper.tsx +37 -37
  180. package/src/lib/components/shared/history-timeline/history-entry.tsx +112 -112
  181. package/src/lib/components/shared/history-timeline/history-note-checkbox.tsx +28 -28
  182. package/src/lib/components/shared/history-timeline/history-note-editor.tsx +60 -60
  183. package/src/lib/components/shared/history-timeline/history-note-input.tsx +39 -39
  184. package/src/lib/components/shared/history-timeline/history-timeline.tsx +56 -56
  185. package/src/lib/components/shared/icon-mark.tsx +18 -18
  186. package/src/lib/components/shared/language-selector.tsx +48 -48
  187. package/src/lib/components/shared/logo-mark.tsx +24 -24
  188. package/src/lib/components/shared/multi-select.tsx +159 -159
  189. package/src/lib/components/shared/option-value-input.tsx +94 -94
  190. package/src/lib/components/shared/paginated-list-data-table.tsx +520 -520
  191. package/src/lib/components/shared/permission-guard.tsx +20 -20
  192. package/src/lib/components/shared/role-code-label.tsx +8 -8
  193. package/src/lib/components/shared/role-selector.tsx +56 -56
  194. package/src/lib/components/shared/seller-selector.tsx +107 -107
  195. package/src/lib/components/shared/tax-category-selector.tsx +65 -65
  196. package/src/lib/components/shared/translatable-form-field.tsx +74 -74
  197. package/src/lib/components/shared/vendure-image.tsx +159 -159
  198. package/src/lib/components/shared/zone-selector.tsx +66 -66
  199. package/src/lib/components/ui/accordion.tsx +59 -59
  200. package/src/lib/components/ui/alert-dialog.tsx +128 -128
  201. package/src/lib/components/ui/alert.tsx +60 -60
  202. package/src/lib/components/ui/avatar.tsx +38 -38
  203. package/src/lib/components/ui/badge.tsx +38 -38
  204. package/src/lib/components/ui/breadcrumb.tsx +102 -102
  205. package/src/lib/components/ui/button.tsx +51 -51
  206. package/src/lib/components/ui/calendar.tsx +69 -69
  207. package/src/lib/components/ui/card.tsx +47 -47
  208. package/src/lib/components/ui/checkbox.tsx +27 -27
  209. package/src/lib/components/ui/collapsible.tsx +33 -33
  210. package/src/lib/components/ui/command.tsx +133 -133
  211. package/src/lib/components/ui/dialog.tsx +116 -116
  212. package/src/lib/components/ui/dropdown-menu.tsx +220 -220
  213. package/src/lib/components/ui/form.tsx +141 -141
  214. package/src/lib/components/ui/hover-card.tsx +36 -36
  215. package/src/lib/components/ui/input.tsx +19 -19
  216. package/src/lib/components/ui/label.tsx +21 -21
  217. package/src/lib/components/ui/pagination.tsx +127 -127
  218. package/src/lib/components/ui/popover.tsx +40 -40
  219. package/src/lib/components/ui/scroll-area.tsx +50 -50
  220. package/src/lib/components/ui/select.tsx +161 -161
  221. package/src/lib/components/ui/separator.tsx +26 -26
  222. package/src/lib/components/ui/sheet.tsx +118 -118
  223. package/src/lib/components/ui/sidebar.tsx +696 -696
  224. package/src/lib/components/ui/skeleton.tsx +13 -13
  225. package/src/lib/components/ui/sonner.tsx +27 -27
  226. package/src/lib/components/ui/switch.tsx +26 -26
  227. package/src/lib/components/ui/table.tsx +82 -82
  228. package/src/lib/components/ui/tabs.tsx +48 -48
  229. package/src/lib/components/ui/textarea.tsx +18 -18
  230. package/src/lib/components/ui/tooltip.tsx +51 -51
  231. package/src/lib/constants.ts +326 -326
  232. package/src/lib/framework/component-registry/component-registry.tsx +70 -70
  233. package/src/lib/framework/component-registry/dynamic-component.tsx +58 -58
  234. package/src/lib/framework/dashboard-widget/base-widget.tsx +97 -97
  235. package/src/lib/framework/dashboard-widget/latest-orders-widget/index.tsx +96 -96
  236. package/src/lib/framework/dashboard-widget/latest-orders-widget/latest-orders-widget.graphql.ts +35 -35
  237. package/src/lib/framework/dashboard-widget/metrics-widget/chart.tsx +24 -24
  238. package/src/lib/framework/dashboard-widget/metrics-widget/index.tsx +82 -82
  239. package/src/lib/framework/dashboard-widget/metrics-widget/metrics-widget.graphql.ts +14 -14
  240. package/src/lib/framework/dashboard-widget/orders-summary/index.tsx +167 -167
  241. package/src/lib/framework/dashboard-widget/orders-summary/order-summary-widget.graphql.ts +14 -14
  242. package/src/lib/framework/dashboard-widget/types.ts +22 -22
  243. package/src/lib/framework/dashboard-widget/widget-extensions.tsx +19 -19
  244. package/src/lib/framework/defaults.ts +219 -219
  245. package/src/lib/framework/document-introspection/add-custom-fields.spec.ts +242 -242
  246. package/src/lib/framework/document-introspection/add-custom-fields.ts +246 -246
  247. package/src/lib/framework/document-introspection/get-document-structure.spec.ts +310 -310
  248. package/src/lib/framework/document-introspection/get-document-structure.ts +460 -460
  249. package/src/lib/framework/document-introspection/hooks.ts +10 -10
  250. package/src/lib/framework/extension-api/define-dashboard-extension.ts +66 -66
  251. package/src/lib/framework/extension-api/extension-api-types.ts +58 -58
  252. package/src/lib/framework/extension-api/use-dashboard-extensions.ts +26 -26
  253. package/src/lib/framework/form-engine/form-schema-tools.ts +98 -98
  254. package/src/lib/framework/form-engine/use-generated-form.tsx +116 -116
  255. package/src/lib/framework/layout-engine/layout-extensions.ts +30 -30
  256. package/src/lib/framework/layout-engine/location-wrapper.tsx +96 -96
  257. package/src/lib/framework/layout-engine/page-layout.tsx +272 -272
  258. package/src/lib/framework/nav-menu/nav-menu-extensions.ts +66 -66
  259. package/src/lib/framework/page/detail-page-route-loader.tsx +48 -48
  260. package/src/lib/framework/page/detail-page.tsx +131 -131
  261. package/src/lib/framework/page/list-page.tsx +166 -166
  262. package/src/lib/framework/page/page-api.ts +9 -9
  263. package/src/lib/framework/page/page-types.ts +51 -51
  264. package/src/lib/framework/page/use-detail-page.ts +217 -217
  265. package/src/lib/framework/page/use-extended-router.tsx +69 -69
  266. package/src/lib/framework/registry/global-registry.ts +46 -46
  267. package/src/lib/framework/registry/registry-types.ts +15 -15
  268. package/src/lib/graphql/api.ts +61 -61
  269. package/src/lib/graphql/fragments.tsx +54 -54
  270. package/src/lib/graphql/graphql-env.d.ts +499 -499
  271. package/src/lib/graphql/graphql.ts +15 -15
  272. package/src/lib/hooks/use-auth.tsx +11 -11
  273. package/src/lib/hooks/use-channel.ts +12 -12
  274. package/src/lib/hooks/use-custom-field-config.ts +10 -10
  275. package/src/lib/hooks/use-grouped-permissions.ts +54 -54
  276. package/src/lib/hooks/use-local-format.ts +119 -119
  277. package/src/lib/hooks/use-mobile.ts +19 -19
  278. package/src/lib/hooks/use-page.tsx +10 -10
  279. package/src/lib/hooks/use-permissions.ts +22 -22
  280. package/src/lib/hooks/use-server-config.ts +4 -4
  281. package/src/lib/hooks/use-theme.ts +10 -10
  282. package/src/lib/hooks/use-user-settings.tsx +12 -12
  283. package/src/lib/index.ts +149 -149
  284. package/src/lib/lib/trans.tsx +16 -16
  285. package/src/lib/lib/utils.ts +60 -60
  286. package/src/lib/providers/auth.tsx +152 -152
  287. package/src/lib/providers/channel-provider.tsx +121 -121
  288. package/src/lib/providers/i18n-provider.tsx +28 -28
  289. package/src/lib/providers/server-config.tsx +279 -279
  290. package/src/lib/providers/theme-provider.tsx +54 -54
  291. package/src/lib/providers/user-settings.tsx +89 -89
  292. package/src/lib/virtual.d.ts +12 -12
  293. package/vite/config-loader.ts +181 -181
  294. package/vite/constants.ts +280 -280
  295. package/vite/index.ts +1 -1
  296. package/vite/schema-generator.ts +40 -40
  297. package/vite/ui-config.ts +60 -60
  298. package/vite/vite-plugin-admin-api-schema.ts +141 -141
  299. package/vite/vite-plugin-config-loader.ts +64 -64
  300. package/vite/vite-plugin-config.ts +42 -42
  301. package/vite/vite-plugin-dashboard-metadata.ts +58 -58
  302. package/vite/vite-plugin-gql-tada.ts +62 -62
  303. package/vite/vite-plugin-theme.ts +195 -195
  304. package/vite/vite-plugin-ui-config.ts +60 -60
  305. package/vite/vite-plugin-vendure-dashboard.ts +118 -118
  306. package/dist/plugin/.vendure-dashboard-temp/dev-config.js +0 -227
  307. package/dist/plugin/.vendure-dashboard-temp/dev-config.js.map +0 -1
  308. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/api-extensions.js +0 -33
  309. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/api-extensions.js.map +0 -1
  310. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/mv.resolver.js +0 -69
  311. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/mv.resolver.js.map +0 -1
  312. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-process.js +0 -110
  313. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-process.js.map +0 -1
  314. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-seller-strategy.js +0 -134
  315. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-seller-strategy.js.map +0 -1
  316. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-payment-handler.js +0 -86
  317. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-payment-handler.js.map +0 -1
  318. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-eligibility-checker.js +0 -49
  319. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-eligibility-checker.js.map +0 -1
  320. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-line-assignment-strategy.js +0 -57
  321. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-line-assignment-strategy.js.map +0 -1
  322. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/constants.js +0 -20
  323. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/constants.js.map +0 -1
  324. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/multivendor.plugin.js +0 -151
  325. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/multivendor.plugin.js.map +0 -1
  326. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/payment/mv-connect-sdk.js +0 -47
  327. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/payment/mv-connect-sdk.js.map +0 -1
  328. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/service/mv.service.js +0 -222
  329. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/service/mv.service.js.map +0 -1
  330. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/types.js +0 -4
  331. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/types.js.map +0 -1
  332. package/dist/plugin/.vendure-dashboard-temp/package.json +0 -3
  333. package/dist/plugin/.vendure-dashboard-temp/schema.graphql +0 -6378
  334. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/api-extensions.js +0 -103
  335. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/api-extensions.js.map +0 -1
  336. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-entity.resolver.js +0 -105
  337. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-entity.resolver.js.map +0 -1
  338. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-admin.resolver.js +0 -183
  339. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-admin.resolver.js.map +0 -1
  340. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-entity.resolver.js +0 -113
  341. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-entity.resolver.js.map +0 -1
  342. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-shop.resolver.js +0 -112
  343. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-shop.resolver.js.map +0 -1
  344. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/entities/product-review.entity.js +0 -111
  345. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/entities/product-review.entity.js.map +0 -1
  346. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-admin-types.js +0 -616
  347. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-admin-types.js.map +0 -1
  348. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-shop-types.js +0 -563
  349. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-shop-types.js.map +0 -1
  350. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/reviews-plugin.js +0 -135
  351. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/reviews-plugin.js.map +0 -1
  352. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/types.js +0 -4
  353. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/types.js.map +0 -1
  354. package/dist/plugin/.vendure-dashboard-temp/tsconfig.json +0 -10
  355. package/dist/plugin/config-loader.d.ts +0 -27
  356. package/dist/plugin/config-loader.js +0 -141
  357. package/dist/plugin/constants.d.ts +0 -5
  358. package/dist/plugin/constants.js +0 -277
  359. package/dist/plugin/index.d.ts +0 -1
  360. package/dist/plugin/index.js +0 -1
  361. package/dist/plugin/schema-generator.d.ts +0 -5
  362. package/dist/plugin/schema-generator.js +0 -24
  363. package/dist/plugin/ui-config.d.ts +0 -3
  364. package/dist/plugin/ui-config.js +0 -35
  365. package/dist/plugin/vite-plugin-admin-api-schema.d.ts +0 -24
  366. package/dist/plugin/vite-plugin-admin-api-schema.js +0 -82
  367. package/dist/plugin/vite-plugin-config-loader.d.ts +0 -17
  368. package/dist/plugin/vite-plugin-config-loader.js +0 -56
  369. package/dist/plugin/vite-plugin-config.d.ts +0 -4
  370. package/dist/plugin/vite-plugin-config.js +0 -38
  371. package/dist/plugin/vite-plugin-dashboard-metadata.d.ts +0 -9
  372. package/dist/plugin/vite-plugin-dashboard-metadata.js +0 -52
  373. package/dist/plugin/vite-plugin-gql-tada.d.ts +0 -6
  374. package/dist/plugin/vite-plugin-gql-tada.js +0 -51
  375. package/dist/plugin/vite-plugin-set-root.d.ts +0 -4
  376. package/dist/plugin/vite-plugin-set-root.js +0 -15
  377. package/dist/plugin/vite-plugin-ui-config.d.ts +0 -15
  378. package/dist/plugin/vite-plugin-ui-config.js +0 -43
  379. package/dist/plugin/vite-plugin-vendure-dashboard.d.ts +0 -35
  380. package/dist/plugin/vite-plugin-vendure-dashboard.js +0 -81
  381. package/src/app/routeTree.gen.ts +0 -1372
@@ -1,520 +1,520 @@
1
- import { DataTableColumnHeader } from '@/components/data-table/data-table-column-header.js';
2
- import { DataTable, FacetedFilter } from '@/components/data-table/data-table.js';
3
- import {
4
- FieldInfo,
5
- getObjectPathToPaginatedList,
6
- getTypeFieldInfo,
7
- } from '@/framework/document-introspection/get-document-structure.js';
8
- import { useListQueryFields } from '@/framework/document-introspection/hooks.js';
9
- import { api } from '@/graphql/api.js';
10
- import { useMutation, useQueryClient } from '@tanstack/react-query';
11
- import { useDebounce } from '@uidotdev/usehooks';
12
-
13
- import {
14
- DropdownMenu,
15
- DropdownMenuContent,
16
- DropdownMenuItem,
17
- DropdownMenuTrigger,
18
- } from '@/components/ui/dropdown-menu.js';
19
- import { DisplayComponent } from '@/framework/component-registry/dynamic-component.js';
20
- import { ResultOf } from '@/graphql/graphql.js';
21
- import { TypedDocumentNode } from '@graphql-typed-document-node/core';
22
- import { Trans, useLingui } from '@/lib/trans.js';
23
- import { useQuery } from '@tanstack/react-query';
24
- import {
25
- ColumnFiltersState,
26
- ColumnSort,
27
- createColumnHelper,
28
- SortingState,
29
- Table,
30
- } from '@tanstack/react-table';
31
- import { AccessorKeyColumnDef, ColumnDef, Row, TableOptions } from '@tanstack/table-core';
32
- import { EllipsisIcon, TrashIcon } from 'lucide-react';
33
- import React, { useMemo } from 'react';
34
- import { toast } from 'sonner';
35
- import { Button } from '../ui/button.js';
36
-
37
- // Type that identifies a paginated list structure (has items array and totalItems)
38
- type IsPaginatedList<T> = T extends { items: any[]; totalItems: number } ? true : false;
39
-
40
- // Helper type to extract string keys from an object
41
- type StringKeys<T> = T extends object ? Extract<keyof T, string> : never;
42
-
43
- // Non-recursive approach to find paginated list paths with max 2 levels
44
- // Level 0: Direct top-level check
45
- type Level0PaginatedLists<T> = T extends object ? (IsPaginatedList<T> extends true ? '' : never) : never;
46
-
47
- // Level 1: One level deep
48
- type Level1PaginatedLists<T> = T extends object
49
- ? {
50
- [K in StringKeys<T>]: NonNullable<T[K]> extends object
51
- ? IsPaginatedList<NonNullable<T[K]>> extends true
52
- ? K
53
- : never
54
- : never;
55
- }[StringKeys<T>]
56
- : never;
57
-
58
- // Level 2: Two levels deep
59
- type Level2PaginatedLists<T> = T extends object
60
- ? {
61
- [K1 in StringKeys<T>]: NonNullable<T[K1]> extends object
62
- ? {
63
- [K2 in StringKeys<NonNullable<T[K1]>>]: NonNullable<NonNullable<T[K1]>[K2]> extends object
64
- ? IsPaginatedList<NonNullable<NonNullable<T[K1]>[K2]>> extends true
65
- ? `${K1}.${K2}`
66
- : never
67
- : never;
68
- }[StringKeys<NonNullable<T[K1]>>]
69
- : never;
70
- }[StringKeys<T>]
71
- : never;
72
-
73
- // Combine all levels
74
- type FindPaginatedListPaths<T> = Level0PaginatedLists<T> | Level1PaginatedLists<T> | Level2PaginatedLists<T>;
75
-
76
- // Extract all paths from a TypedDocumentNode
77
- export type PaginatedListPaths<T extends TypedDocumentNode<any, any>> =
78
- FindPaginatedListPaths<ResultOf<T>> extends infer Paths ? (Paths extends '' ? never : Paths) : never;
79
-
80
- export type PaginatedListItemFields<
81
- T extends TypedDocumentNode<any, any>,
82
- Path extends PaginatedListPaths<T> = PaginatedListPaths<T>,
83
- > =
84
- // split the path by '.' if it exists
85
- Path extends `${infer First}.${infer Rest}`
86
- ? NonNullable<ResultOf<T>[First]>[Rest]['items'][number]
87
- : Path extends keyof ResultOf<T>
88
- ? ResultOf<T>[Path] extends { items: Array<infer Item> }
89
- ? ResultOf<T>[Path]['items'][number]
90
- : never
91
- : never;
92
-
93
- export type PaginatedListKeys<
94
- T extends TypedDocumentNode<any, any>,
95
- Path extends PaginatedListPaths<T> = PaginatedListPaths<T>,
96
- > = {
97
- [K in keyof PaginatedListItemFields<T, Path>]: K;
98
- }[keyof PaginatedListItemFields<T, Path>];
99
-
100
- export type CustomizeColumnConfig<T extends TypedDocumentNode<any, any>> = {
101
- [Key in keyof PaginatedListItemFields<T>]?: Partial<
102
- ColumnDef<PaginatedListItemFields<T>, PaginatedListItemFields<T>[Key]>
103
- >;
104
- };
105
-
106
- export type FacetedFilterConfig<T extends TypedDocumentNode<any, any>> = {
107
- [Key in keyof PaginatedListItemFields<T>]?: FacetedFilter;
108
- };
109
-
110
- export type ListQueryShape =
111
- | {
112
- [key: string]: {
113
- items: any[];
114
- totalItems: number;
115
- };
116
- }
117
- | {
118
- [key: string]: {
119
- [key: string]: {
120
- items: any[];
121
- totalItems: number;
122
- };
123
- };
124
- };
125
-
126
- export type ListQueryOptionsShape = {
127
- options?: {
128
- skip?: number;
129
- take?: number;
130
- sort?: {
131
- [key: string]: 'ASC' | 'DESC';
132
- };
133
- filter?: any;
134
- };
135
- [key: string]: any;
136
- };
137
-
138
- export type AdditionalColumns<T extends TypedDocumentNode<any, any>> = {
139
- [key: string]: ColumnDef<PaginatedListItemFields<T>>;
140
- };
141
-
142
- export interface PaginatedListContext {
143
- refetchPaginatedList: () => void;
144
- }
145
-
146
- export const PaginatedListContext = React.createContext<PaginatedListContext | undefined>(undefined);
147
-
148
- /**
149
- * @description
150
- * Returns the context for the paginated list data table. Must be used within a PaginatedListDataTable.
151
- *
152
- * @example
153
- * ```ts
154
- * const { refetchPaginatedList } = usePaginatedList();
155
- *
156
- * const mutation = useMutation({
157
- * mutationFn: api.mutate(updateFacetValueDocument),
158
- * onSuccess: () => {
159
- * refetchPaginatedList();
160
- * },
161
- * });
162
- * ```
163
- */
164
- export function usePaginatedList() {
165
- const context = React.useContext(PaginatedListContext);
166
- if (!context) {
167
- throw new Error('usePaginatedList must be used within a PaginatedListDataTable');
168
- }
169
- return context;
170
- }
171
-
172
- export interface RowAction<T> {
173
- label: React.ReactNode;
174
- onClick?: (row: Row<T>) => void;
175
- }
176
-
177
- export interface PaginatedListDataTableProps<
178
- T extends TypedDocumentNode<U, V>,
179
- U extends any,
180
- V extends ListQueryOptionsShape,
181
- AC extends AdditionalColumns<T>,
182
- > {
183
- listQuery: T;
184
- deleteMutation?: TypedDocumentNode<any, any>;
185
- transformQueryKey?: (queryKey: any[]) => any[];
186
- transformVariables?: (variables: V) => V;
187
- customizeColumns?: CustomizeColumnConfig<T>;
188
- additionalColumns?: AC;
189
- defaultColumnOrder?: (keyof PaginatedListItemFields<T> | AC[number]['id'])[];
190
- defaultVisibility?: Partial<Record<keyof PaginatedListItemFields<T>, boolean>>;
191
- onSearchTermChange?: (searchTerm: string) => NonNullable<V['options']>['filter'];
192
- page: number;
193
- itemsPerPage: number;
194
- sorting: SortingState;
195
- columnFilters?: ColumnFiltersState;
196
- onPageChange: (table: Table<any>, page: number, perPage: number) => void;
197
- onSortChange: (table: Table<any>, sorting: SortingState) => void;
198
- onFilterChange: (table: Table<any>, filters: ColumnFiltersState) => void;
199
- facetedFilters?: FacetedFilterConfig<T>;
200
- rowActions?: RowAction<PaginatedListItemFields<T>>[];
201
- disableViewOptions?: boolean;
202
- transformData?: (data: PaginatedListItemFields<T>[]) => PaginatedListItemFields<T>[];
203
- setTableOptions?: (table: TableOptions<any>) => TableOptions<any>;
204
- }
205
-
206
- export const PaginatedListDataTableKey = 'PaginatedListDataTable';
207
-
208
- export function PaginatedListDataTable<
209
- T extends TypedDocumentNode<U, V>,
210
- U extends Record<string, any> = any,
211
- V extends ListQueryOptionsShape = any,
212
- AC extends AdditionalColumns<T> = AdditionalColumns<T>,
213
- >({
214
- listQuery,
215
- deleteMutation,
216
- transformQueryKey,
217
- transformVariables,
218
- customizeColumns,
219
- additionalColumns,
220
- defaultVisibility,
221
- defaultColumnOrder,
222
- onSearchTermChange,
223
- page,
224
- itemsPerPage,
225
- sorting,
226
- columnFilters,
227
- onPageChange,
228
- onSortChange,
229
- onFilterChange,
230
- facetedFilters,
231
- rowActions,
232
- disableViewOptions,
233
- setTableOptions,
234
- transformData,
235
- }: PaginatedListDataTableProps<T, U, V, AC>) {
236
- const [searchTerm, setSearchTerm] = React.useState<string>('');
237
- const debouncedSearchTerm = useDebounce(searchTerm, 500);
238
- const queryClient = useQueryClient();
239
-
240
- const sort = sorting?.reduce((acc: any, sort: ColumnSort) => {
241
- const direction = sort.desc ? 'DESC' : 'ASC';
242
- const field = sort.id;
243
-
244
- if (!field || !direction) {
245
- return acc;
246
- }
247
- return { ...acc, [field]: direction };
248
- }, {});
249
-
250
- const filter = columnFilters?.length
251
- ? {
252
- _and: columnFilters.map(f => {
253
- if (Array.isArray(f.value)) {
254
- return { [f.id]: { in: f.value } };
255
- }
256
- return { [f.id]: f.value };
257
- }),
258
- }
259
- : undefined;
260
-
261
- const defaultQueryKey = [
262
- PaginatedListDataTableKey,
263
- listQuery,
264
- page,
265
- itemsPerPage,
266
- sorting,
267
- filter,
268
- debouncedSearchTerm,
269
- ];
270
- const queryKey = transformQueryKey ? transformQueryKey(defaultQueryKey) : defaultQueryKey;
271
-
272
- function refetchPaginatedList() {
273
- queryClient.invalidateQueries({ queryKey });
274
- }
275
-
276
- const { data } = useQuery({
277
- queryFn: () => {
278
- const searchFilter = onSearchTermChange ? onSearchTermChange(debouncedSearchTerm) : {};
279
- const mergedFilter = { ...filter, ...searchFilter };
280
- const variables = {
281
- options: {
282
- take: itemsPerPage,
283
- skip: (page - 1) * itemsPerPage,
284
- sort,
285
- filter: mergedFilter,
286
- },
287
- } as V;
288
-
289
- const transformedVariables = transformVariables ? transformVariables(variables) : variables;
290
- return api.query(listQuery, transformedVariables);
291
- },
292
- queryKey,
293
- });
294
-
295
- const fields = useListQueryFields(listQuery);
296
- const paginatedListObjectPath = getObjectPathToPaginatedList(listQuery);
297
-
298
- let listData = data as any;
299
- for (const path of paginatedListObjectPath) {
300
- listData = listData?.[path];
301
- }
302
-
303
- const columnHelper = createColumnHelper<PaginatedListItemFields<T>>();
304
-
305
- const { columns, customFieldColumnNames } = useMemo(() => {
306
- const columnConfigs: Array<{ fieldInfo: FieldInfo; isCustomField: boolean }> = [];
307
- const customFieldColumnNames: string[] = [];
308
-
309
- columnConfigs.push(
310
- ...fields // Filter out custom fields
311
- .filter(field => field.name !== 'customFields' && !field.type.endsWith('CustomFields'))
312
- .map(field => ({ fieldInfo: field, isCustomField: false })),
313
- );
314
-
315
- const customFieldColumn = fields.find(field => field.name === 'customFields');
316
- if (customFieldColumn && customFieldColumn.type !== 'JSON') {
317
- const customFieldFields = getTypeFieldInfo(customFieldColumn.type);
318
- columnConfigs.push(
319
- ...customFieldFields.map(field => ({ fieldInfo: field, isCustomField: true })),
320
- );
321
- customFieldColumnNames.push(...customFieldFields.map(field => field.name));
322
- }
323
-
324
- const queryBasedColumns = columnConfigs.map(({ fieldInfo, isCustomField }) => {
325
- const customConfig = customizeColumns?.[fieldInfo.name as keyof PaginatedListItemFields<T>] ?? {};
326
- const { header, ...customConfigRest } = customConfig;
327
- const enableColumnFilter = fieldInfo.isScalar && !facetedFilters?.[fieldInfo.name];
328
-
329
- return columnHelper.accessor(fieldInfo.name as any, {
330
- id: fieldInfo.name,
331
- meta: { fieldInfo, isCustomField },
332
- enableColumnFilter,
333
- enableSorting: fieldInfo.isScalar,
334
- cell: ({ cell, row }) => {
335
- const value = !isCustomField
336
- ? cell.getValue()
337
- : (row.original as any)?.customFields?.[fieldInfo.name];
338
- if (fieldInfo.list && Array.isArray(value)) {
339
- return value.join(', ');
340
- }
341
- if (
342
- (fieldInfo.type === 'DateTime' && typeof value === 'string') ||
343
- value instanceof Date
344
- ) {
345
- return <DisplayComponent id="vendure:dateTime" value={value} />;
346
- }
347
- if (fieldInfo.type === 'Boolean') {
348
- if (cell.column.id === 'enabled') {
349
- return <DisplayComponent id="vendure:booleanBadge" value={value} />;
350
- } else {
351
- return <DisplayComponent id="vendure:booleanCheckbox" value={value} />;
352
- }
353
- }
354
- if (fieldInfo.type === 'Asset') {
355
- return <DisplayComponent id="vendure:asset" value={value} />;
356
- }
357
- if (value !== null && typeof value === 'object') {
358
- return JSON.stringify(value);
359
- }
360
- return value;
361
- },
362
- header: headerContext => {
363
- return (
364
- <DataTableColumnHeader headerContext={headerContext} customConfig={customConfig} />
365
- );
366
- },
367
- ...customConfigRest,
368
- });
369
- });
370
-
371
- let finalColumns = [...queryBasedColumns];
372
-
373
- for (const [id, column] of Object.entries(additionalColumns ?? {})) {
374
- if (!id) {
375
- throw new Error('Column id is required');
376
- }
377
- finalColumns.push(columnHelper.accessor(id as any, { ...column, id }));
378
- }
379
-
380
- if (defaultColumnOrder) {
381
- // ensure the columns with ids matching the items in defaultColumnOrder
382
- // appear as the first columns in sequence, and leave the remainder in the
383
- // existing order
384
- const orderedColumns = finalColumns
385
- .filter(column => column.id && defaultColumnOrder.includes(column.id))
386
- .sort((a, b) => defaultColumnOrder.indexOf(a.id) - defaultColumnOrder.indexOf(b.id));
387
- const remainingColumns = finalColumns.filter(
388
- column => !column.id || !defaultColumnOrder.includes(column.id),
389
- );
390
- finalColumns = [...orderedColumns, ...remainingColumns];
391
- }
392
-
393
- if (rowActions || deleteMutation) {
394
- const rowActionColumn = getRowActions(rowActions, deleteMutation);
395
- if (rowActionColumn) {
396
- finalColumns.push(rowActionColumn);
397
- }
398
- }
399
-
400
- return { columns: finalColumns, customFieldColumnNames };
401
- }, [fields, customizeColumns, rowActions]);
402
-
403
- const columnVisibility = getColumnVisibility(fields, defaultVisibility, customFieldColumnNames);
404
- const transformedData =
405
- typeof transformData === 'function' ? transformData(listData?.items ?? []) : (listData?.items ?? []);
406
- return (
407
- <PaginatedListContext.Provider value={{ refetchPaginatedList }}>
408
- <DataTable
409
- columns={columns}
410
- data={transformedData}
411
- page={page}
412
- itemsPerPage={itemsPerPage}
413
- sorting={sorting}
414
- columnFilters={columnFilters}
415
- totalItems={listData?.totalItems ?? 0}
416
- onPageChange={onPageChange}
417
- onSortChange={onSortChange}
418
- onFilterChange={onFilterChange}
419
- onSearchTermChange={onSearchTermChange ? term => setSearchTerm(term) : undefined}
420
- defaultColumnVisibility={columnVisibility}
421
- facetedFilters={facetedFilters}
422
- disableViewOptions={disableViewOptions}
423
- setTableOptions={setTableOptions}
424
- />
425
- </PaginatedListContext.Provider>
426
- );
427
- }
428
-
429
- function getRowActions(
430
- rowActions?: RowAction<any>[],
431
- deleteMutation?: TypedDocumentNode<any, any>,
432
- ): AccessorKeyColumnDef<any> | undefined {
433
- return {
434
- id: 'actions',
435
- accessorKey: 'actions',
436
- header: 'Actions',
437
- cell: ({ row }) => {
438
- return (
439
- <DropdownMenu>
440
- <DropdownMenuTrigger asChild>
441
- <Button variant="ghost" size="icon">
442
- <EllipsisIcon />
443
- </Button>
444
- </DropdownMenuTrigger>
445
- <DropdownMenuContent>
446
- {rowActions?.map((action, index) => (
447
- <DropdownMenuItem onClick={() => action.onClick?.(row)} key={index}>
448
- {action.label}
449
- </DropdownMenuItem>
450
- ))}
451
- {deleteMutation && (
452
- <DeleteMutationRowAction deleteMutation={deleteMutation} row={row} />
453
- )}
454
- </DropdownMenuContent>
455
- </DropdownMenu>
456
- );
457
- },
458
- };
459
- }
460
-
461
- function DeleteMutationRowAction({
462
- deleteMutation,
463
- row,
464
- }: {
465
- deleteMutation: TypedDocumentNode<any, any>;
466
- row: Row<{ id: string }>;
467
- }) {
468
- const { refetchPaginatedList } = usePaginatedList();
469
- const { i18n } = useLingui();
470
- const { mutate: deleteMutationFn } = useMutation({
471
- mutationFn: api.mutate(deleteMutation),
472
- onSuccess: (result: { [key: string]: { result: 'DELETED' | 'NOT_DELETED'; message: string } }) => {
473
- const unwrappedResult = Object.values(result)[0];
474
- if (unwrappedResult.result === 'DELETED') {
475
- refetchPaginatedList();
476
- toast.success(i18n.t('Deleted successfully'));
477
- } else {
478
- toast.error(i18n.t('Failed to delete'), {
479
- description: unwrappedResult.message,
480
- });
481
- }
482
- },
483
- onError: (err: Error) => {
484
- toast.error(i18n.t('Failed to delete'), {
485
- description: err.message,
486
- });
487
- },
488
- });
489
- return (
490
- <DropdownMenuItem onClick={() => deleteMutationFn({ id: row.original.id })}>
491
- <div className="flex items-center gap-2 text-destructive">
492
- <TrashIcon className="w-4 h-4 text-destructive" />
493
- <Trans>Delete</Trans>
494
- </div>
495
- </DropdownMenuItem>
496
- );
497
- }
498
- /**
499
- * Returns the default column visibility configuration.
500
- */
501
- function getColumnVisibility(
502
- fields: FieldInfo[],
503
- defaultVisibility?: Record<string, boolean | undefined>,
504
- customFieldColumnNames?: string[],
505
- ): Record<string, boolean> {
506
- const allDefaultsTrue = defaultVisibility && Object.values(defaultVisibility).every(v => v === true);
507
- const allDefaultsFalse = defaultVisibility && Object.values(defaultVisibility).every(v => v === false);
508
- return {
509
- id: false,
510
- createdAt: false,
511
- updatedAt: false,
512
- ...(allDefaultsTrue ? { ...Object.fromEntries(fields.map(f => [f.name, false])) } : {}),
513
- ...(allDefaultsFalse ? { ...Object.fromEntries(fields.map(f => [f.name, true])) } : {}),
514
- ...defaultVisibility,
515
- // Make custom fields hidden by default
516
- ...(customFieldColumnNames
517
- ? { ...Object.fromEntries(customFieldColumnNames.map(f => [f, false])) }
518
- : {}),
519
- };
520
- }
1
+ import { DataTableColumnHeader } from '@/components/data-table/data-table-column-header.js';
2
+ import { DataTable, FacetedFilter } from '@/components/data-table/data-table.js';
3
+ import {
4
+ FieldInfo,
5
+ getObjectPathToPaginatedList,
6
+ getTypeFieldInfo,
7
+ } from '@/framework/document-introspection/get-document-structure.js';
8
+ import { useListQueryFields } from '@/framework/document-introspection/hooks.js';
9
+ import { api } from '@/graphql/api.js';
10
+ import { useMutation, useQueryClient } from '@tanstack/react-query';
11
+ import { useDebounce } from '@uidotdev/usehooks';
12
+
13
+ import {
14
+ DropdownMenu,
15
+ DropdownMenuContent,
16
+ DropdownMenuItem,
17
+ DropdownMenuTrigger,
18
+ } from '@/components/ui/dropdown-menu.js';
19
+ import { DisplayComponent } from '@/framework/component-registry/dynamic-component.js';
20
+ import { ResultOf } from '@/graphql/graphql.js';
21
+ import { TypedDocumentNode } from '@graphql-typed-document-node/core';
22
+ import { Trans, useLingui } from '@/lib/trans.js';
23
+ import { useQuery } from '@tanstack/react-query';
24
+ import {
25
+ ColumnFiltersState,
26
+ ColumnSort,
27
+ createColumnHelper,
28
+ SortingState,
29
+ Table,
30
+ } from '@tanstack/react-table';
31
+ import { AccessorKeyColumnDef, ColumnDef, Row, TableOptions } from '@tanstack/table-core';
32
+ import { EllipsisIcon, TrashIcon } from 'lucide-react';
33
+ import React, { useMemo } from 'react';
34
+ import { toast } from 'sonner';
35
+ import { Button } from '../ui/button.js';
36
+
37
+ // Type that identifies a paginated list structure (has items array and totalItems)
38
+ type IsPaginatedList<T> = T extends { items: any[]; totalItems: number } ? true : false;
39
+
40
+ // Helper type to extract string keys from an object
41
+ type StringKeys<T> = T extends object ? Extract<keyof T, string> : never;
42
+
43
+ // Non-recursive approach to find paginated list paths with max 2 levels
44
+ // Level 0: Direct top-level check
45
+ type Level0PaginatedLists<T> = T extends object ? (IsPaginatedList<T> extends true ? '' : never) : never;
46
+
47
+ // Level 1: One level deep
48
+ type Level1PaginatedLists<T> = T extends object
49
+ ? {
50
+ [K in StringKeys<T>]: NonNullable<T[K]> extends object
51
+ ? IsPaginatedList<NonNullable<T[K]>> extends true
52
+ ? K
53
+ : never
54
+ : never;
55
+ }[StringKeys<T>]
56
+ : never;
57
+
58
+ // Level 2: Two levels deep
59
+ type Level2PaginatedLists<T> = T extends object
60
+ ? {
61
+ [K1 in StringKeys<T>]: NonNullable<T[K1]> extends object
62
+ ? {
63
+ [K2 in StringKeys<NonNullable<T[K1]>>]: NonNullable<NonNullable<T[K1]>[K2]> extends object
64
+ ? IsPaginatedList<NonNullable<NonNullable<T[K1]>[K2]>> extends true
65
+ ? `${K1}.${K2}`
66
+ : never
67
+ : never;
68
+ }[StringKeys<NonNullable<T[K1]>>]
69
+ : never;
70
+ }[StringKeys<T>]
71
+ : never;
72
+
73
+ // Combine all levels
74
+ type FindPaginatedListPaths<T> = Level0PaginatedLists<T> | Level1PaginatedLists<T> | Level2PaginatedLists<T>;
75
+
76
+ // Extract all paths from a TypedDocumentNode
77
+ export type PaginatedListPaths<T extends TypedDocumentNode<any, any>> =
78
+ FindPaginatedListPaths<ResultOf<T>> extends infer Paths ? (Paths extends '' ? never : Paths) : never;
79
+
80
+ export type PaginatedListItemFields<
81
+ T extends TypedDocumentNode<any, any>,
82
+ Path extends PaginatedListPaths<T> = PaginatedListPaths<T>,
83
+ > =
84
+ // split the path by '.' if it exists
85
+ Path extends `${infer First}.${infer Rest}`
86
+ ? NonNullable<ResultOf<T>[First]>[Rest]['items'][number]
87
+ : Path extends keyof ResultOf<T>
88
+ ? ResultOf<T>[Path] extends { items: Array<infer Item> }
89
+ ? ResultOf<T>[Path]['items'][number]
90
+ : never
91
+ : never;
92
+
93
+ export type PaginatedListKeys<
94
+ T extends TypedDocumentNode<any, any>,
95
+ Path extends PaginatedListPaths<T> = PaginatedListPaths<T>,
96
+ > = {
97
+ [K in keyof PaginatedListItemFields<T, Path>]: K;
98
+ }[keyof PaginatedListItemFields<T, Path>];
99
+
100
+ export type CustomizeColumnConfig<T extends TypedDocumentNode<any, any>> = {
101
+ [Key in keyof PaginatedListItemFields<T>]?: Partial<
102
+ ColumnDef<PaginatedListItemFields<T>, PaginatedListItemFields<T>[Key]>
103
+ >;
104
+ };
105
+
106
+ export type FacetedFilterConfig<T extends TypedDocumentNode<any, any>> = {
107
+ [Key in keyof PaginatedListItemFields<T>]?: FacetedFilter;
108
+ };
109
+
110
+ export type ListQueryShape =
111
+ | {
112
+ [key: string]: {
113
+ items: any[];
114
+ totalItems: number;
115
+ };
116
+ }
117
+ | {
118
+ [key: string]: {
119
+ [key: string]: {
120
+ items: any[];
121
+ totalItems: number;
122
+ };
123
+ };
124
+ };
125
+
126
+ export type ListQueryOptionsShape = {
127
+ options?: {
128
+ skip?: number;
129
+ take?: number;
130
+ sort?: {
131
+ [key: string]: 'ASC' | 'DESC';
132
+ };
133
+ filter?: any;
134
+ };
135
+ [key: string]: any;
136
+ };
137
+
138
+ export type AdditionalColumns<T extends TypedDocumentNode<any, any>> = {
139
+ [key: string]: ColumnDef<PaginatedListItemFields<T>>;
140
+ };
141
+
142
+ export interface PaginatedListContext {
143
+ refetchPaginatedList: () => void;
144
+ }
145
+
146
+ export const PaginatedListContext = React.createContext<PaginatedListContext | undefined>(undefined);
147
+
148
+ /**
149
+ * @description
150
+ * Returns the context for the paginated list data table. Must be used within a PaginatedListDataTable.
151
+ *
152
+ * @example
153
+ * ```ts
154
+ * const { refetchPaginatedList } = usePaginatedList();
155
+ *
156
+ * const mutation = useMutation({
157
+ * mutationFn: api.mutate(updateFacetValueDocument),
158
+ * onSuccess: () => {
159
+ * refetchPaginatedList();
160
+ * },
161
+ * });
162
+ * ```
163
+ */
164
+ export function usePaginatedList() {
165
+ const context = React.useContext(PaginatedListContext);
166
+ if (!context) {
167
+ throw new Error('usePaginatedList must be used within a PaginatedListDataTable');
168
+ }
169
+ return context;
170
+ }
171
+
172
+ export interface RowAction<T> {
173
+ label: React.ReactNode;
174
+ onClick?: (row: Row<T>) => void;
175
+ }
176
+
177
+ export interface PaginatedListDataTableProps<
178
+ T extends TypedDocumentNode<U, V>,
179
+ U extends any,
180
+ V extends ListQueryOptionsShape,
181
+ AC extends AdditionalColumns<T>,
182
+ > {
183
+ listQuery: T;
184
+ deleteMutation?: TypedDocumentNode<any, any>;
185
+ transformQueryKey?: (queryKey: any[]) => any[];
186
+ transformVariables?: (variables: V) => V;
187
+ customizeColumns?: CustomizeColumnConfig<T>;
188
+ additionalColumns?: AC;
189
+ defaultColumnOrder?: (keyof PaginatedListItemFields<T> | AC[number]['id'])[];
190
+ defaultVisibility?: Partial<Record<keyof PaginatedListItemFields<T>, boolean>>;
191
+ onSearchTermChange?: (searchTerm: string) => NonNullable<V['options']>['filter'];
192
+ page: number;
193
+ itemsPerPage: number;
194
+ sorting: SortingState;
195
+ columnFilters?: ColumnFiltersState;
196
+ onPageChange: (table: Table<any>, page: number, perPage: number) => void;
197
+ onSortChange: (table: Table<any>, sorting: SortingState) => void;
198
+ onFilterChange: (table: Table<any>, filters: ColumnFiltersState) => void;
199
+ facetedFilters?: FacetedFilterConfig<T>;
200
+ rowActions?: RowAction<PaginatedListItemFields<T>>[];
201
+ disableViewOptions?: boolean;
202
+ transformData?: (data: PaginatedListItemFields<T>[]) => PaginatedListItemFields<T>[];
203
+ setTableOptions?: (table: TableOptions<any>) => TableOptions<any>;
204
+ }
205
+
206
+ export const PaginatedListDataTableKey = 'PaginatedListDataTable';
207
+
208
+ export function PaginatedListDataTable<
209
+ T extends TypedDocumentNode<U, V>,
210
+ U extends Record<string, any> = any,
211
+ V extends ListQueryOptionsShape = any,
212
+ AC extends AdditionalColumns<T> = AdditionalColumns<T>,
213
+ >({
214
+ listQuery,
215
+ deleteMutation,
216
+ transformQueryKey,
217
+ transformVariables,
218
+ customizeColumns,
219
+ additionalColumns,
220
+ defaultVisibility,
221
+ defaultColumnOrder,
222
+ onSearchTermChange,
223
+ page,
224
+ itemsPerPage,
225
+ sorting,
226
+ columnFilters,
227
+ onPageChange,
228
+ onSortChange,
229
+ onFilterChange,
230
+ facetedFilters,
231
+ rowActions,
232
+ disableViewOptions,
233
+ setTableOptions,
234
+ transformData,
235
+ }: PaginatedListDataTableProps<T, U, V, AC>) {
236
+ const [searchTerm, setSearchTerm] = React.useState<string>('');
237
+ const debouncedSearchTerm = useDebounce(searchTerm, 500);
238
+ const queryClient = useQueryClient();
239
+
240
+ const sort = sorting?.reduce((acc: any, sort: ColumnSort) => {
241
+ const direction = sort.desc ? 'DESC' : 'ASC';
242
+ const field = sort.id;
243
+
244
+ if (!field || !direction) {
245
+ return acc;
246
+ }
247
+ return { ...acc, [field]: direction };
248
+ }, {});
249
+
250
+ const filter = columnFilters?.length
251
+ ? {
252
+ _and: columnFilters.map(f => {
253
+ if (Array.isArray(f.value)) {
254
+ return { [f.id]: { in: f.value } };
255
+ }
256
+ return { [f.id]: f.value };
257
+ }),
258
+ }
259
+ : undefined;
260
+
261
+ const defaultQueryKey = [
262
+ PaginatedListDataTableKey,
263
+ listQuery,
264
+ page,
265
+ itemsPerPage,
266
+ sorting,
267
+ filter,
268
+ debouncedSearchTerm,
269
+ ];
270
+ const queryKey = transformQueryKey ? transformQueryKey(defaultQueryKey) : defaultQueryKey;
271
+
272
+ function refetchPaginatedList() {
273
+ queryClient.invalidateQueries({ queryKey });
274
+ }
275
+
276
+ const { data } = useQuery({
277
+ queryFn: () => {
278
+ const searchFilter = onSearchTermChange ? onSearchTermChange(debouncedSearchTerm) : {};
279
+ const mergedFilter = { ...filter, ...searchFilter };
280
+ const variables = {
281
+ options: {
282
+ take: itemsPerPage,
283
+ skip: (page - 1) * itemsPerPage,
284
+ sort,
285
+ filter: mergedFilter,
286
+ },
287
+ } as V;
288
+
289
+ const transformedVariables = transformVariables ? transformVariables(variables) : variables;
290
+ return api.query(listQuery, transformedVariables);
291
+ },
292
+ queryKey,
293
+ });
294
+
295
+ const fields = useListQueryFields(listQuery);
296
+ const paginatedListObjectPath = getObjectPathToPaginatedList(listQuery);
297
+
298
+ let listData = data as any;
299
+ for (const path of paginatedListObjectPath) {
300
+ listData = listData?.[path];
301
+ }
302
+
303
+ const columnHelper = createColumnHelper<PaginatedListItemFields<T>>();
304
+
305
+ const { columns, customFieldColumnNames } = useMemo(() => {
306
+ const columnConfigs: Array<{ fieldInfo: FieldInfo; isCustomField: boolean }> = [];
307
+ const customFieldColumnNames: string[] = [];
308
+
309
+ columnConfigs.push(
310
+ ...fields // Filter out custom fields
311
+ .filter(field => field.name !== 'customFields' && !field.type.endsWith('CustomFields'))
312
+ .map(field => ({ fieldInfo: field, isCustomField: false })),
313
+ );
314
+
315
+ const customFieldColumn = fields.find(field => field.name === 'customFields');
316
+ if (customFieldColumn && customFieldColumn.type !== 'JSON') {
317
+ const customFieldFields = getTypeFieldInfo(customFieldColumn.type);
318
+ columnConfigs.push(
319
+ ...customFieldFields.map(field => ({ fieldInfo: field, isCustomField: true })),
320
+ );
321
+ customFieldColumnNames.push(...customFieldFields.map(field => field.name));
322
+ }
323
+
324
+ const queryBasedColumns = columnConfigs.map(({ fieldInfo, isCustomField }) => {
325
+ const customConfig = customizeColumns?.[fieldInfo.name as keyof PaginatedListItemFields<T>] ?? {};
326
+ const { header, ...customConfigRest } = customConfig;
327
+ const enableColumnFilter = fieldInfo.isScalar && !facetedFilters?.[fieldInfo.name];
328
+
329
+ return columnHelper.accessor(fieldInfo.name as any, {
330
+ id: fieldInfo.name,
331
+ meta: { fieldInfo, isCustomField },
332
+ enableColumnFilter,
333
+ enableSorting: fieldInfo.isScalar,
334
+ cell: ({ cell, row }) => {
335
+ const value = !isCustomField
336
+ ? cell.getValue()
337
+ : (row.original as any)?.customFields?.[fieldInfo.name];
338
+ if (fieldInfo.list && Array.isArray(value)) {
339
+ return value.join(', ');
340
+ }
341
+ if (
342
+ (fieldInfo.type === 'DateTime' && typeof value === 'string') ||
343
+ value instanceof Date
344
+ ) {
345
+ return <DisplayComponent id="vendure:dateTime" value={value} />;
346
+ }
347
+ if (fieldInfo.type === 'Boolean') {
348
+ if (cell.column.id === 'enabled') {
349
+ return <DisplayComponent id="vendure:booleanBadge" value={value} />;
350
+ } else {
351
+ return <DisplayComponent id="vendure:booleanCheckbox" value={value} />;
352
+ }
353
+ }
354
+ if (fieldInfo.type === 'Asset') {
355
+ return <DisplayComponent id="vendure:asset" value={value} />;
356
+ }
357
+ if (value !== null && typeof value === 'object') {
358
+ return JSON.stringify(value);
359
+ }
360
+ return value;
361
+ },
362
+ header: headerContext => {
363
+ return (
364
+ <DataTableColumnHeader headerContext={headerContext} customConfig={customConfig} />
365
+ );
366
+ },
367
+ ...customConfigRest,
368
+ });
369
+ });
370
+
371
+ let finalColumns = [...queryBasedColumns];
372
+
373
+ for (const [id, column] of Object.entries(additionalColumns ?? {})) {
374
+ if (!id) {
375
+ throw new Error('Column id is required');
376
+ }
377
+ finalColumns.push(columnHelper.accessor(id as any, { ...column, id }));
378
+ }
379
+
380
+ if (defaultColumnOrder) {
381
+ // ensure the columns with ids matching the items in defaultColumnOrder
382
+ // appear as the first columns in sequence, and leave the remainder in the
383
+ // existing order
384
+ const orderedColumns = finalColumns
385
+ .filter(column => column.id && defaultColumnOrder.includes(column.id))
386
+ .sort((a, b) => defaultColumnOrder.indexOf(a.id) - defaultColumnOrder.indexOf(b.id));
387
+ const remainingColumns = finalColumns.filter(
388
+ column => !column.id || !defaultColumnOrder.includes(column.id),
389
+ );
390
+ finalColumns = [...orderedColumns, ...remainingColumns];
391
+ }
392
+
393
+ if (rowActions || deleteMutation) {
394
+ const rowActionColumn = getRowActions(rowActions, deleteMutation);
395
+ if (rowActionColumn) {
396
+ finalColumns.push(rowActionColumn);
397
+ }
398
+ }
399
+
400
+ return { columns: finalColumns, customFieldColumnNames };
401
+ }, [fields, customizeColumns, rowActions]);
402
+
403
+ const columnVisibility = getColumnVisibility(fields, defaultVisibility, customFieldColumnNames);
404
+ const transformedData =
405
+ typeof transformData === 'function' ? transformData(listData?.items ?? []) : (listData?.items ?? []);
406
+ return (
407
+ <PaginatedListContext.Provider value={{ refetchPaginatedList }}>
408
+ <DataTable
409
+ columns={columns}
410
+ data={transformedData}
411
+ page={page}
412
+ itemsPerPage={itemsPerPage}
413
+ sorting={sorting}
414
+ columnFilters={columnFilters}
415
+ totalItems={listData?.totalItems ?? 0}
416
+ onPageChange={onPageChange}
417
+ onSortChange={onSortChange}
418
+ onFilterChange={onFilterChange}
419
+ onSearchTermChange={onSearchTermChange ? term => setSearchTerm(term) : undefined}
420
+ defaultColumnVisibility={columnVisibility}
421
+ facetedFilters={facetedFilters}
422
+ disableViewOptions={disableViewOptions}
423
+ setTableOptions={setTableOptions}
424
+ />
425
+ </PaginatedListContext.Provider>
426
+ );
427
+ }
428
+
429
+ function getRowActions(
430
+ rowActions?: RowAction<any>[],
431
+ deleteMutation?: TypedDocumentNode<any, any>,
432
+ ): AccessorKeyColumnDef<any> | undefined {
433
+ return {
434
+ id: 'actions',
435
+ accessorKey: 'actions',
436
+ header: 'Actions',
437
+ cell: ({ row }) => {
438
+ return (
439
+ <DropdownMenu>
440
+ <DropdownMenuTrigger asChild>
441
+ <Button variant="ghost" size="icon">
442
+ <EllipsisIcon />
443
+ </Button>
444
+ </DropdownMenuTrigger>
445
+ <DropdownMenuContent>
446
+ {rowActions?.map((action, index) => (
447
+ <DropdownMenuItem onClick={() => action.onClick?.(row)} key={index}>
448
+ {action.label}
449
+ </DropdownMenuItem>
450
+ ))}
451
+ {deleteMutation && (
452
+ <DeleteMutationRowAction deleteMutation={deleteMutation} row={row} />
453
+ )}
454
+ </DropdownMenuContent>
455
+ </DropdownMenu>
456
+ );
457
+ },
458
+ };
459
+ }
460
+
461
+ function DeleteMutationRowAction({
462
+ deleteMutation,
463
+ row,
464
+ }: {
465
+ deleteMutation: TypedDocumentNode<any, any>;
466
+ row: Row<{ id: string }>;
467
+ }) {
468
+ const { refetchPaginatedList } = usePaginatedList();
469
+ const { i18n } = useLingui();
470
+ const { mutate: deleteMutationFn } = useMutation({
471
+ mutationFn: api.mutate(deleteMutation),
472
+ onSuccess: (result: { [key: string]: { result: 'DELETED' | 'NOT_DELETED'; message: string } }) => {
473
+ const unwrappedResult = Object.values(result)[0];
474
+ if (unwrappedResult.result === 'DELETED') {
475
+ refetchPaginatedList();
476
+ toast.success(i18n.t('Deleted successfully'));
477
+ } else {
478
+ toast.error(i18n.t('Failed to delete'), {
479
+ description: unwrappedResult.message,
480
+ });
481
+ }
482
+ },
483
+ onError: (err: Error) => {
484
+ toast.error(i18n.t('Failed to delete'), {
485
+ description: err.message,
486
+ });
487
+ },
488
+ });
489
+ return (
490
+ <DropdownMenuItem onClick={() => deleteMutationFn({ id: row.original.id })}>
491
+ <div className="flex items-center gap-2 text-destructive">
492
+ <TrashIcon className="w-4 h-4 text-destructive" />
493
+ <Trans>Delete</Trans>
494
+ </div>
495
+ </DropdownMenuItem>
496
+ );
497
+ }
498
+ /**
499
+ * Returns the default column visibility configuration.
500
+ */
501
+ function getColumnVisibility(
502
+ fields: FieldInfo[],
503
+ defaultVisibility?: Record<string, boolean | undefined>,
504
+ customFieldColumnNames?: string[],
505
+ ): Record<string, boolean> {
506
+ const allDefaultsTrue = defaultVisibility && Object.values(defaultVisibility).every(v => v === true);
507
+ const allDefaultsFalse = defaultVisibility && Object.values(defaultVisibility).every(v => v === false);
508
+ return {
509
+ id: false,
510
+ createdAt: false,
511
+ updatedAt: false,
512
+ ...(allDefaultsTrue ? { ...Object.fromEntries(fields.map(f => [f.name, false])) } : {}),
513
+ ...(allDefaultsFalse ? { ...Object.fromEntries(fields.map(f => [f.name, true])) } : {}),
514
+ ...defaultVisibility,
515
+ // Make custom fields hidden by default
516
+ ...(customFieldColumnNames
517
+ ? { ...Object.fromEntries(customFieldColumnNames.map(f => [f, false])) }
518
+ : {}),
519
+ };
520
+ }