@vendure/dashboard 3.2.0 → 3.2.2

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 (364) hide show
  1. package/LICENSE.md +42 -0
  2. package/README.md +12 -12
  3. package/dist/plugin/vite-plugin-admin-api-schema.js +2 -2
  4. package/dist/plugin/vite-plugin-dashboard-metadata.js +4 -4
  5. package/dist/plugin/vite-plugin-theme.d.ts +55 -0
  6. package/dist/plugin/vite-plugin-theme.js +130 -0
  7. package/dist/plugin/vite-plugin-ui-config.js +2 -2
  8. package/dist/plugin/vite-plugin-vendure-dashboard.d.ts +2 -1
  9. package/dist/plugin/vite-plugin-vendure-dashboard.js +2 -0
  10. package/index.html +15 -15
  11. package/lingui.config.js +12 -12
  12. package/package.json +15 -5
  13. package/src/app/app-providers.tsx +30 -30
  14. package/src/app/main.tsx +97 -97
  15. package/src/app/routes/__root.tsx +24 -24
  16. package/src/app/routes/_authenticated/_administrators/administrators.graphql.ts +79 -79
  17. package/src/app/routes/_authenticated/_administrators/administrators.tsx +86 -86
  18. package/src/app/routes/_authenticated/_administrators/administrators_.$id.tsx +155 -155
  19. package/src/app/routes/_authenticated/_administrators/components/role-permissions-display.tsx +133 -133
  20. package/src/app/routes/_authenticated/_assets/assets.tsx +19 -19
  21. package/src/app/routes/_authenticated/_channels/channels.graphql.ts +93 -93
  22. package/src/app/routes/_authenticated/_channels/channels.tsx +60 -60
  23. package/src/app/routes/_authenticated/_channels/channels_.$id.tsx +248 -248
  24. package/src/app/routes/_authenticated/_collections/collections.graphql.ts +133 -133
  25. package/src/app/routes/_authenticated/_collections/collections.tsx +195 -195
  26. package/src/app/routes/_authenticated/_collections/collections_.$id.tsx +224 -224
  27. package/src/app/routes/_authenticated/_collections/components/collection-contents-preview-table.tsx +127 -127
  28. package/src/app/routes/_authenticated/_collections/components/collection-contents-sheet.tsx +46 -46
  29. package/src/app/routes/_authenticated/_collections/components/collection-contents-table.tsx +82 -82
  30. package/src/app/routes/_authenticated/_collections/components/collection-filters-selector.tsx +91 -91
  31. package/src/app/routes/_authenticated/_countries/countries.graphql.ts +69 -69
  32. package/src/app/routes/_authenticated/_countries/countries.tsx +67 -67
  33. package/src/app/routes/_authenticated/_countries/countries_.$id.tsx +122 -122
  34. package/src/app/routes/_authenticated/_customer-groups/components/customer-group-members-sheet.tsx +44 -44
  35. package/src/app/routes/_authenticated/_customer-groups/components/customer-group-members-table.tsx +129 -129
  36. package/src/app/routes/_authenticated/_customer-groups/customer-groups.graphql.ts +71 -71
  37. package/src/app/routes/_authenticated/_customer-groups/customer-groups.tsx +68 -68
  38. package/src/app/routes/_authenticated/_customer-groups/customer-groups_.$id.tsx +111 -111
  39. package/src/app/routes/_authenticated/_customers/components/customer-address-card.tsx +155 -155
  40. package/src/app/routes/_authenticated/_customers/components/customer-address-form.tsx +344 -344
  41. package/src/app/routes/_authenticated/_customers/components/customer-group-controls.tsx +4 -4
  42. package/src/app/routes/_authenticated/_customers/components/customer-history/customer-history-container.tsx +78 -78
  43. package/src/app/routes/_authenticated/_customers/components/customer-history/customer-history.tsx +77 -77
  44. package/src/app/routes/_authenticated/_customers/components/customer-history/index.ts +3 -3
  45. package/src/app/routes/_authenticated/_customers/components/customer-history/use-customer-history.ts +169 -169
  46. package/src/app/routes/_authenticated/_customers/components/customer-order-table.tsx +88 -88
  47. package/src/app/routes/_authenticated/_customers/components/customer-status-badge.tsx +33 -33
  48. package/src/app/routes/_authenticated/_customers/customers.graphql.ts +204 -204
  49. package/src/app/routes/_authenticated/_customers/customers.tsx +82 -82
  50. package/src/app/routes/_authenticated/_customers/customers_.$id.tsx +274 -274
  51. package/src/app/routes/_authenticated/_facets/components/edit-facet-value.tsx +129 -129
  52. package/src/app/routes/_authenticated/_facets/components/facet-values-sheet.tsx +46 -46
  53. package/src/app/routes/_authenticated/_facets/components/facet-values-table.tsx +97 -97
  54. package/src/app/routes/_authenticated/_facets/facets.graphql.ts +104 -104
  55. package/src/app/routes/_authenticated/_facets/facets.tsx +97 -97
  56. package/src/app/routes/_authenticated/_facets/facets_.$id.tsx +139 -139
  57. package/src/app/routes/_authenticated/_global-settings/global-settings.graphql.ts +28 -28
  58. package/src/app/routes/_authenticated/_global-settings/global-settings.tsx +161 -161
  59. package/src/app/routes/_authenticated/_orders/components/order-address.tsx +58 -58
  60. package/src/app/routes/_authenticated/_orders/components/order-history/index.ts +3 -3
  61. package/src/app/routes/_authenticated/_orders/components/order-history/order-history-container.tsx +72 -72
  62. package/src/app/routes/_authenticated/_orders/components/order-history/order-history.tsx +96 -96
  63. package/src/app/routes/_authenticated/_orders/components/order-history/use-order-history.ts +171 -171
  64. package/src/app/routes/_authenticated/_orders/components/order-table.tsx +169 -169
  65. package/src/app/routes/_authenticated/_orders/components/order-tax-summary.tsx +38 -38
  66. package/src/app/routes/_authenticated/_orders/components/payment-details.tsx +61 -61
  67. package/src/app/routes/_authenticated/_orders/orders.graphql.ts +325 -325
  68. package/src/app/routes/_authenticated/_orders/orders.tsx +120 -120
  69. package/src/app/routes/_authenticated/_orders/orders_.$id.tsx +133 -133
  70. package/src/app/routes/_authenticated/_payment-methods/components/payment-eligibility-checker-selector.tsx +104 -104
  71. package/src/app/routes/_authenticated/_payment-methods/components/payment-handler-selector.tsx +100 -100
  72. package/src/app/routes/_authenticated/_payment-methods/payment-methods.graphql.ts +83 -83
  73. package/src/app/routes/_authenticated/_payment-methods/payment-methods.tsx +64 -64
  74. package/src/app/routes/_authenticated/_payment-methods/payment-methods_.$id.tsx +183 -183
  75. package/src/app/routes/_authenticated/_product-variants/components/variant-price-detail.tsx +87 -87
  76. package/src/app/routes/_authenticated/_product-variants/product-variants.graphql.ts +123 -123
  77. package/src/app/routes/_authenticated/_product-variants/product-variants.tsx +78 -78
  78. package/src/app/routes/_authenticated/_product-variants/product-variants_.$id.tsx +331 -331
  79. package/src/app/routes/_authenticated/_products/components/create-product-variants-dialog.tsx +228 -228
  80. package/src/app/routes/_authenticated/_products/components/create-product-variants.tsx +462 -462
  81. package/src/app/routes/_authenticated/_products/components/option-value-input.tsx +95 -95
  82. package/src/app/routes/_authenticated/_products/components/product-variants-table.tsx +87 -87
  83. package/src/app/routes/_authenticated/_products/products.graphql.ts +116 -116
  84. package/src/app/routes/_authenticated/_products/products.tsx +48 -48
  85. package/src/app/routes/_authenticated/_products/products_.$id.tsx +196 -196
  86. package/src/app/routes/_authenticated/_profile/profile.graphql.ts +23 -23
  87. package/src/app/routes/_authenticated/_profile/profile.tsx +122 -122
  88. package/src/app/routes/_authenticated/_promotions/components/promotion-actions-selector.tsx +107 -107
  89. package/src/app/routes/_authenticated/_promotions/components/promotion-conditions-selector.tsx +107 -107
  90. package/src/app/routes/_authenticated/_promotions/promotions.graphql.ts +96 -96
  91. package/src/app/routes/_authenticated/_promotions/promotions.tsx +61 -61
  92. package/src/app/routes/_authenticated/_promotions/promotions_.$id.tsx +235 -235
  93. package/src/app/routes/_authenticated/_roles/components/expandable-permissions.tsx +54 -54
  94. package/src/app/routes/_authenticated/_roles/components/permissions-grid.tsx +116 -116
  95. package/src/app/routes/_authenticated/_roles/roles.graphql.ts +67 -67
  96. package/src/app/routes/_authenticated/_roles/roles.tsx +96 -96
  97. package/src/app/routes/_authenticated/_roles/roles_.$id.tsx +142 -142
  98. package/src/app/routes/_authenticated/_sellers/sellers.graphql.ts +61 -61
  99. package/src/app/routes/_authenticated/_sellers/sellers.tsx +51 -51
  100. package/src/app/routes/_authenticated/_sellers/sellers_.$id.tsx +111 -111
  101. package/src/app/routes/_authenticated/_shipping-methods/components/fulfillment-handler-selector.tsx +56 -56
  102. package/src/app/routes/_authenticated/_shipping-methods/components/shipping-calculator-selector.tsx +101 -101
  103. package/src/app/routes/_authenticated/_shipping-methods/components/shipping-eligibility-checker-selector.tsx +101 -101
  104. package/src/app/routes/_authenticated/_shipping-methods/components/test-shipping-method-dialog.tsx +32 -32
  105. package/src/app/routes/_authenticated/_shipping-methods/shipping-methods.graphql.ts +83 -83
  106. package/src/app/routes/_authenticated/_shipping-methods/shipping-methods.tsx +55 -55
  107. package/src/app/routes/_authenticated/_shipping-methods/shipping-methods_.$id.tsx +171 -171
  108. package/src/app/routes/_authenticated/_stock-locations/stock-locations.graphql.ts +62 -62
  109. package/src/app/routes/_authenticated/_stock-locations/stock-locations.tsx +48 -48
  110. package/src/app/routes/_authenticated/_stock-locations/stock-locations_.$id.tsx +115 -115
  111. package/src/app/routes/_authenticated/_system/components/payload-dialog.tsx +34 -34
  112. package/src/app/routes/_authenticated/_system/healthchecks.tsx +93 -93
  113. package/src/app/routes/_authenticated/_system/job-queue.graphql.ts +43 -43
  114. package/src/app/routes/_authenticated/_system/job-queue.tsx +161 -161
  115. package/src/app/routes/_authenticated/_tax-categories/tax-categories.graphql.ts +63 -63
  116. package/src/app/routes/_authenticated/_tax-categories/tax-categories.tsx +65 -65
  117. package/src/app/routes/_authenticated/_tax-categories/tax-categories_.$id.tsx +115 -115
  118. package/src/app/routes/_authenticated/_tax-rates/tax-rates.graphql.ts +75 -75
  119. package/src/app/routes/_authenticated/_tax-rates/tax-rates.tsx +108 -108
  120. package/src/app/routes/_authenticated/_tax-rates/tax-rates_.$id.tsx +148 -148
  121. package/src/app/routes/_authenticated/_zones/components/zone-countries-sheet.tsx +31 -31
  122. package/src/app/routes/_authenticated/_zones/components/zone-countries-table.tsx +79 -79
  123. package/src/app/routes/_authenticated/_zones/zones.graphql.ts +96 -96
  124. package/src/app/routes/_authenticated/_zones/zones.tsx +57 -57
  125. package/src/app/routes/_authenticated/_zones/zones_.$id.tsx +103 -103
  126. package/src/app/routes/_authenticated/index.tsx +194 -194
  127. package/src/app/routes/_authenticated.tsx +25 -25
  128. package/src/app/routes/login.tsx +48 -48
  129. package/src/app/styles.css +82 -82
  130. package/src/app/tailwindcss-animate.css +275 -275
  131. package/src/i18n/locales/de.po +1579 -1579
  132. package/src/i18n/locales/en.po +1579 -1579
  133. package/src/lib/components/data-display/boolean.tsx +23 -23
  134. package/src/lib/components/data-display/date-time.tsx +13 -13
  135. package/src/lib/components/data-display/json.tsx +5 -5
  136. package/src/lib/components/data-display/money.tsx +15 -15
  137. package/src/lib/components/data-input/affixed-input.tsx +49 -49
  138. package/src/lib/components/data-input/customer-group-input.tsx +72 -72
  139. package/src/lib/components/data-input/datetime-input.tsx +149 -149
  140. package/src/lib/components/data-input/facet-value-input.tsx +68 -68
  141. package/src/lib/components/data-input/money-input.tsx +112 -112
  142. package/src/lib/components/data-input/richt-text-input.tsx +99 -99
  143. package/src/lib/components/data-table/data-table-column-header.tsx +73 -73
  144. package/src/lib/components/data-table/data-table-faceted-filter.tsx +172 -172
  145. package/src/lib/components/data-table/data-table-filter-dialog.tsx +73 -73
  146. package/src/lib/components/data-table/data-table-pagination.tsx +87 -87
  147. package/src/lib/components/data-table/data-table-view-options.tsx +53 -53
  148. package/src/lib/components/data-table/data-table.tsx +218 -218
  149. package/src/lib/components/layout/app-layout.tsx +42 -42
  150. package/src/lib/components/layout/app-sidebar.tsx +34 -34
  151. package/src/lib/components/layout/channel-switcher.tsx +87 -87
  152. package/src/lib/components/layout/content-language-selector.tsx +41 -41
  153. package/src/lib/components/layout/generated-breadcrumbs.tsx +82 -82
  154. package/src/lib/components/layout/language-dialog.tsx +104 -104
  155. package/src/lib/components/layout/nav-main.tsx +178 -178
  156. package/src/lib/components/layout/nav-projects.tsx +81 -81
  157. package/src/lib/components/layout/nav-user.tsx +176 -176
  158. package/src/lib/components/layout/prerelease-popup.tsx +38 -38
  159. package/src/lib/components/login/login-form.tsx +173 -173
  160. package/src/lib/components/shared/alerts.tsx +20 -20
  161. package/src/lib/components/shared/animated-number.tsx +49 -49
  162. package/src/lib/components/shared/asset-gallery.tsx +433 -433
  163. package/src/lib/components/shared/asset-picker-dialog.tsx +71 -71
  164. package/src/lib/components/shared/asset-preview-dialog.tsx +48 -48
  165. package/src/lib/components/shared/asset-preview.tsx +345 -345
  166. package/src/lib/components/shared/assigned-facet-values.tsx +68 -68
  167. package/src/lib/components/shared/channel-code-label.tsx +7 -7
  168. package/src/lib/components/shared/channel-selector.tsx +51 -51
  169. package/src/lib/components/shared/configurable-operation-arg-input.tsx +51 -51
  170. package/src/lib/components/shared/configurable-operation-input.tsx +133 -133
  171. package/src/lib/components/shared/confirmation-dialog.tsx +58 -58
  172. package/src/lib/components/shared/copyable-text.tsx +31 -31
  173. package/src/lib/components/shared/country-selector.tsx +105 -105
  174. package/src/lib/components/shared/currency-selector.tsx +33 -33
  175. package/src/lib/components/shared/custom-fields-form.tsx +86 -86
  176. package/src/lib/components/shared/customer-address-form.tsx +330 -330
  177. package/src/lib/components/shared/customer-group-chip.tsx +30 -30
  178. package/src/lib/components/shared/customer-group-selector.tsx +62 -62
  179. package/src/lib/components/shared/customer-selector.tsx +107 -107
  180. package/src/lib/components/shared/detail-page-button.tsx +22 -22
  181. package/src/lib/components/shared/entity-assets.tsx +340 -340
  182. package/src/lib/components/shared/error-page.tsx +31 -31
  183. package/src/lib/components/shared/facet-value-chip.tsx +44 -44
  184. package/src/lib/components/shared/facet-value-selector.tsx +306 -306
  185. package/src/lib/components/shared/focal-point-control.tsx +64 -64
  186. package/src/lib/components/shared/form-field-wrapper.tsx +37 -37
  187. package/src/lib/components/shared/history-timeline/history-entry.tsx +112 -112
  188. package/src/lib/components/shared/history-timeline/history-note-checkbox.tsx +28 -28
  189. package/src/lib/components/shared/history-timeline/history-note-editor.tsx +60 -60
  190. package/src/lib/components/shared/history-timeline/history-note-input.tsx +39 -39
  191. package/src/lib/components/shared/history-timeline/history-timeline.tsx +56 -56
  192. package/src/lib/components/shared/icon-mark.tsx +18 -18
  193. package/src/lib/components/shared/language-selector.tsx +48 -48
  194. package/src/lib/components/shared/logo-mark.tsx +24 -24
  195. package/src/lib/components/shared/multi-select.tsx +159 -159
  196. package/src/lib/components/shared/option-value-input.tsx +94 -94
  197. package/src/lib/components/shared/paginated-list-data-table.tsx +520 -520
  198. package/src/lib/components/shared/permission-guard.tsx +20 -20
  199. package/src/lib/components/shared/role-code-label.tsx +8 -8
  200. package/src/lib/components/shared/role-selector.tsx +56 -56
  201. package/src/lib/components/shared/seller-selector.tsx +107 -107
  202. package/src/lib/components/shared/tax-category-selector.tsx +65 -65
  203. package/src/lib/components/shared/translatable-form-field.tsx +74 -74
  204. package/src/lib/components/shared/vendure-image.tsx +159 -159
  205. package/src/lib/components/shared/zone-selector.tsx +66 -66
  206. package/src/lib/components/ui/accordion.tsx +59 -59
  207. package/src/lib/components/ui/alert-dialog.tsx +128 -128
  208. package/src/lib/components/ui/alert.tsx +60 -60
  209. package/src/lib/components/ui/avatar.tsx +38 -38
  210. package/src/lib/components/ui/badge.tsx +38 -38
  211. package/src/lib/components/ui/breadcrumb.tsx +102 -102
  212. package/src/lib/components/ui/button.tsx +51 -51
  213. package/src/lib/components/ui/calendar.tsx +69 -69
  214. package/src/lib/components/ui/card.tsx +47 -47
  215. package/src/lib/components/ui/checkbox.tsx +27 -27
  216. package/src/lib/components/ui/collapsible.tsx +33 -33
  217. package/src/lib/components/ui/command.tsx +133 -133
  218. package/src/lib/components/ui/dialog.tsx +116 -116
  219. package/src/lib/components/ui/dropdown-menu.tsx +220 -220
  220. package/src/lib/components/ui/form.tsx +141 -141
  221. package/src/lib/components/ui/hover-card.tsx +36 -36
  222. package/src/lib/components/ui/input.tsx +19 -19
  223. package/src/lib/components/ui/label.tsx +21 -21
  224. package/src/lib/components/ui/pagination.tsx +127 -127
  225. package/src/lib/components/ui/popover.tsx +40 -40
  226. package/src/lib/components/ui/scroll-area.tsx +50 -50
  227. package/src/lib/components/ui/select.tsx +161 -161
  228. package/src/lib/components/ui/separator.tsx +26 -26
  229. package/src/lib/components/ui/sheet.tsx +118 -118
  230. package/src/lib/components/ui/sidebar.tsx +696 -696
  231. package/src/lib/components/ui/skeleton.tsx +13 -13
  232. package/src/lib/components/ui/sonner.tsx +27 -27
  233. package/src/lib/components/ui/switch.tsx +26 -26
  234. package/src/lib/components/ui/table.tsx +82 -82
  235. package/src/lib/components/ui/tabs.tsx +48 -48
  236. package/src/lib/components/ui/textarea.tsx +18 -18
  237. package/src/lib/components/ui/tooltip.tsx +51 -51
  238. package/src/lib/constants.ts +326 -326
  239. package/src/lib/framework/component-registry/component-registry.tsx +70 -70
  240. package/src/lib/framework/component-registry/dynamic-component.tsx +58 -58
  241. package/src/lib/framework/dashboard-widget/base-widget.tsx +97 -97
  242. package/src/lib/framework/dashboard-widget/latest-orders-widget/index.tsx +96 -96
  243. package/src/lib/framework/dashboard-widget/latest-orders-widget/latest-orders-widget.graphql.ts +35 -35
  244. package/src/lib/framework/dashboard-widget/metrics-widget/chart.tsx +24 -24
  245. package/src/lib/framework/dashboard-widget/metrics-widget/index.tsx +82 -82
  246. package/src/lib/framework/dashboard-widget/metrics-widget/metrics-widget.graphql.ts +14 -14
  247. package/src/lib/framework/dashboard-widget/orders-summary/index.tsx +167 -167
  248. package/src/lib/framework/dashboard-widget/orders-summary/order-summary-widget.graphql.ts +14 -14
  249. package/src/lib/framework/dashboard-widget/types.ts +22 -22
  250. package/src/lib/framework/dashboard-widget/widget-extensions.tsx +19 -19
  251. package/src/lib/framework/defaults.ts +219 -219
  252. package/src/lib/framework/document-introspection/add-custom-fields.spec.ts +242 -242
  253. package/src/lib/framework/document-introspection/add-custom-fields.ts +246 -246
  254. package/src/lib/framework/document-introspection/get-document-structure.spec.ts +310 -310
  255. package/src/lib/framework/document-introspection/get-document-structure.ts +460 -460
  256. package/src/lib/framework/document-introspection/hooks.ts +10 -10
  257. package/src/lib/framework/extension-api/define-dashboard-extension.ts +66 -66
  258. package/src/lib/framework/extension-api/extension-api-types.ts +58 -58
  259. package/src/lib/framework/extension-api/use-dashboard-extensions.ts +26 -26
  260. package/src/lib/framework/form-engine/form-schema-tools.ts +98 -98
  261. package/src/lib/framework/form-engine/use-generated-form.tsx +116 -116
  262. package/src/lib/framework/layout-engine/layout-extensions.ts +30 -30
  263. package/src/lib/framework/layout-engine/location-wrapper.tsx +96 -96
  264. package/src/lib/framework/layout-engine/page-layout.tsx +272 -272
  265. package/src/lib/framework/nav-menu/nav-menu-extensions.ts +66 -66
  266. package/src/lib/framework/page/detail-page-route-loader.tsx +48 -48
  267. package/src/lib/framework/page/detail-page.tsx +131 -131
  268. package/src/lib/framework/page/list-page.tsx +166 -166
  269. package/src/lib/framework/page/page-api.ts +9 -9
  270. package/src/lib/framework/page/page-types.ts +51 -51
  271. package/src/lib/framework/page/use-detail-page.ts +217 -217
  272. package/src/lib/framework/page/use-extended-router.tsx +69 -69
  273. package/src/lib/framework/registry/global-registry.ts +46 -46
  274. package/src/lib/framework/registry/registry-types.ts +15 -15
  275. package/src/lib/graphql/api.ts +61 -61
  276. package/src/lib/graphql/fragments.tsx +54 -54
  277. package/src/lib/graphql/graphql-env.d.ts +499 -499
  278. package/src/lib/graphql/graphql.ts +15 -15
  279. package/src/lib/hooks/use-auth.tsx +11 -11
  280. package/src/lib/hooks/use-channel.ts +12 -12
  281. package/src/lib/hooks/use-custom-field-config.ts +10 -10
  282. package/src/lib/hooks/use-grouped-permissions.ts +54 -54
  283. package/src/lib/hooks/use-local-format.ts +119 -119
  284. package/src/lib/hooks/use-mobile.ts +19 -19
  285. package/src/lib/hooks/use-page.tsx +10 -10
  286. package/src/lib/hooks/use-permissions.ts +22 -22
  287. package/src/lib/hooks/use-server-config.ts +4 -4
  288. package/src/lib/hooks/use-theme.ts +10 -10
  289. package/src/lib/hooks/use-user-settings.tsx +12 -12
  290. package/src/lib/index.ts +149 -149
  291. package/src/lib/lib/trans.tsx +16 -16
  292. package/src/lib/lib/utils.ts +60 -60
  293. package/src/lib/providers/auth.tsx +152 -152
  294. package/src/lib/providers/channel-provider.tsx +121 -121
  295. package/src/lib/providers/i18n-provider.tsx +28 -28
  296. package/src/lib/providers/server-config.tsx +279 -279
  297. package/src/lib/providers/theme-provider.tsx +54 -54
  298. package/src/lib/providers/user-settings.tsx +89 -89
  299. package/src/lib/virtual.d.ts +12 -12
  300. package/vite/config-loader.ts +181 -181
  301. package/vite/constants.ts +280 -280
  302. package/vite/index.ts +1 -1
  303. package/vite/schema-generator.ts +40 -40
  304. package/vite/ui-config.ts +60 -60
  305. package/vite/vite-plugin-admin-api-schema.ts +141 -141
  306. package/vite/vite-plugin-config-loader.ts +64 -64
  307. package/vite/vite-plugin-config.ts +42 -42
  308. package/vite/vite-plugin-dashboard-metadata.ts +58 -58
  309. package/vite/vite-plugin-gql-tada.ts +62 -62
  310. package/vite/vite-plugin-theme.ts +195 -195
  311. package/vite/vite-plugin-ui-config.ts +60 -60
  312. package/vite/vite-plugin-vendure-dashboard.ts +118 -118
  313. package/dist/plugin/.vendure-dashboard-temp/dev-config.js +0 -227
  314. package/dist/plugin/.vendure-dashboard-temp/dev-config.js.map +0 -1
  315. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/api-extensions.js +0 -33
  316. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/api-extensions.js.map +0 -1
  317. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/mv.resolver.js +0 -69
  318. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/mv.resolver.js.map +0 -1
  319. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-process.js +0 -110
  320. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-process.js.map +0 -1
  321. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-seller-strategy.js +0 -134
  322. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-seller-strategy.js.map +0 -1
  323. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-payment-handler.js +0 -86
  324. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-payment-handler.js.map +0 -1
  325. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-eligibility-checker.js +0 -49
  326. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-eligibility-checker.js.map +0 -1
  327. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-line-assignment-strategy.js +0 -57
  328. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-line-assignment-strategy.js.map +0 -1
  329. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/constants.js +0 -20
  330. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/constants.js.map +0 -1
  331. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/multivendor.plugin.js +0 -151
  332. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/multivendor.plugin.js.map +0 -1
  333. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/payment/mv-connect-sdk.js +0 -47
  334. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/payment/mv-connect-sdk.js.map +0 -1
  335. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/service/mv.service.js +0 -222
  336. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/service/mv.service.js.map +0 -1
  337. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/types.js +0 -4
  338. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/types.js.map +0 -1
  339. package/dist/plugin/.vendure-dashboard-temp/package.json +0 -3
  340. package/dist/plugin/.vendure-dashboard-temp/schema.graphql +0 -6378
  341. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/api-extensions.js +0 -103
  342. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/api-extensions.js.map +0 -1
  343. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-entity.resolver.js +0 -105
  344. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-entity.resolver.js.map +0 -1
  345. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-admin.resolver.js +0 -183
  346. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-admin.resolver.js.map +0 -1
  347. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-entity.resolver.js +0 -113
  348. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-entity.resolver.js.map +0 -1
  349. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-shop.resolver.js +0 -112
  350. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-shop.resolver.js.map +0 -1
  351. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/entities/product-review.entity.js +0 -111
  352. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/entities/product-review.entity.js.map +0 -1
  353. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-admin-types.js +0 -616
  354. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-admin-types.js.map +0 -1
  355. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-shop-types.js +0 -563
  356. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-shop-types.js.map +0 -1
  357. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/reviews-plugin.js +0 -135
  358. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/reviews-plugin.js.map +0 -1
  359. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/types.js +0 -4
  360. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/types.js.map +0 -1
  361. package/dist/plugin/.vendure-dashboard-temp/tsconfig.json +0 -10
  362. package/dist/plugin/vite-plugin-set-root.d.ts +0 -4
  363. package/dist/plugin/vite-plugin-set-root.js +0 -15
  364. 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
+ }