@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.
- package/LICENSE.md +42 -0
- package/README.md +12 -12
- package/dist/plugin/vite-plugin-admin-api-schema.js +2 -2
- package/dist/plugin/vite-plugin-dashboard-metadata.js +4 -4
- package/dist/plugin/vite-plugin-theme.d.ts +55 -0
- package/dist/plugin/vite-plugin-theme.js +130 -0
- package/dist/plugin/vite-plugin-ui-config.js +2 -2
- package/dist/plugin/vite-plugin-vendure-dashboard.d.ts +2 -1
- package/dist/plugin/vite-plugin-vendure-dashboard.js +2 -0
- package/index.html +15 -15
- package/lingui.config.js +12 -12
- package/package.json +15 -5
- package/src/app/app-providers.tsx +30 -30
- package/src/app/main.tsx +97 -97
- package/src/app/routes/__root.tsx +24 -24
- package/src/app/routes/_authenticated/_administrators/administrators.graphql.ts +79 -79
- package/src/app/routes/_authenticated/_administrators/administrators.tsx +86 -86
- package/src/app/routes/_authenticated/_administrators/administrators_.$id.tsx +155 -155
- package/src/app/routes/_authenticated/_administrators/components/role-permissions-display.tsx +133 -133
- package/src/app/routes/_authenticated/_assets/assets.tsx +19 -19
- package/src/app/routes/_authenticated/_channels/channels.graphql.ts +93 -93
- package/src/app/routes/_authenticated/_channels/channels.tsx +60 -60
- package/src/app/routes/_authenticated/_channels/channels_.$id.tsx +248 -248
- package/src/app/routes/_authenticated/_collections/collections.graphql.ts +133 -133
- package/src/app/routes/_authenticated/_collections/collections.tsx +195 -195
- package/src/app/routes/_authenticated/_collections/collections_.$id.tsx +224 -224
- package/src/app/routes/_authenticated/_collections/components/collection-contents-preview-table.tsx +127 -127
- package/src/app/routes/_authenticated/_collections/components/collection-contents-sheet.tsx +46 -46
- package/src/app/routes/_authenticated/_collections/components/collection-contents-table.tsx +82 -82
- package/src/app/routes/_authenticated/_collections/components/collection-filters-selector.tsx +91 -91
- package/src/app/routes/_authenticated/_countries/countries.graphql.ts +69 -69
- package/src/app/routes/_authenticated/_countries/countries.tsx +67 -67
- package/src/app/routes/_authenticated/_countries/countries_.$id.tsx +122 -122
- package/src/app/routes/_authenticated/_customer-groups/components/customer-group-members-sheet.tsx +44 -44
- package/src/app/routes/_authenticated/_customer-groups/components/customer-group-members-table.tsx +129 -129
- package/src/app/routes/_authenticated/_customer-groups/customer-groups.graphql.ts +71 -71
- package/src/app/routes/_authenticated/_customer-groups/customer-groups.tsx +68 -68
- package/src/app/routes/_authenticated/_customer-groups/customer-groups_.$id.tsx +111 -111
- package/src/app/routes/_authenticated/_customers/components/customer-address-card.tsx +155 -155
- package/src/app/routes/_authenticated/_customers/components/customer-address-form.tsx +344 -344
- package/src/app/routes/_authenticated/_customers/components/customer-group-controls.tsx +4 -4
- package/src/app/routes/_authenticated/_customers/components/customer-history/customer-history-container.tsx +78 -78
- package/src/app/routes/_authenticated/_customers/components/customer-history/customer-history.tsx +77 -77
- package/src/app/routes/_authenticated/_customers/components/customer-history/index.ts +3 -3
- package/src/app/routes/_authenticated/_customers/components/customer-history/use-customer-history.ts +169 -169
- package/src/app/routes/_authenticated/_customers/components/customer-order-table.tsx +88 -88
- package/src/app/routes/_authenticated/_customers/components/customer-status-badge.tsx +33 -33
- package/src/app/routes/_authenticated/_customers/customers.graphql.ts +204 -204
- package/src/app/routes/_authenticated/_customers/customers.tsx +82 -82
- package/src/app/routes/_authenticated/_customers/customers_.$id.tsx +274 -274
- package/src/app/routes/_authenticated/_facets/components/edit-facet-value.tsx +129 -129
- package/src/app/routes/_authenticated/_facets/components/facet-values-sheet.tsx +46 -46
- package/src/app/routes/_authenticated/_facets/components/facet-values-table.tsx +97 -97
- package/src/app/routes/_authenticated/_facets/facets.graphql.ts +104 -104
- package/src/app/routes/_authenticated/_facets/facets.tsx +97 -97
- package/src/app/routes/_authenticated/_facets/facets_.$id.tsx +139 -139
- package/src/app/routes/_authenticated/_global-settings/global-settings.graphql.ts +28 -28
- package/src/app/routes/_authenticated/_global-settings/global-settings.tsx +161 -161
- package/src/app/routes/_authenticated/_orders/components/order-address.tsx +58 -58
- package/src/app/routes/_authenticated/_orders/components/order-history/index.ts +3 -3
- package/src/app/routes/_authenticated/_orders/components/order-history/order-history-container.tsx +72 -72
- package/src/app/routes/_authenticated/_orders/components/order-history/order-history.tsx +96 -96
- package/src/app/routes/_authenticated/_orders/components/order-history/use-order-history.ts +171 -171
- package/src/app/routes/_authenticated/_orders/components/order-table.tsx +169 -169
- package/src/app/routes/_authenticated/_orders/components/order-tax-summary.tsx +38 -38
- package/src/app/routes/_authenticated/_orders/components/payment-details.tsx +61 -61
- package/src/app/routes/_authenticated/_orders/orders.graphql.ts +325 -325
- package/src/app/routes/_authenticated/_orders/orders.tsx +120 -120
- package/src/app/routes/_authenticated/_orders/orders_.$id.tsx +133 -133
- package/src/app/routes/_authenticated/_payment-methods/components/payment-eligibility-checker-selector.tsx +104 -104
- package/src/app/routes/_authenticated/_payment-methods/components/payment-handler-selector.tsx +100 -100
- package/src/app/routes/_authenticated/_payment-methods/payment-methods.graphql.ts +83 -83
- package/src/app/routes/_authenticated/_payment-methods/payment-methods.tsx +64 -64
- package/src/app/routes/_authenticated/_payment-methods/payment-methods_.$id.tsx +183 -183
- package/src/app/routes/_authenticated/_product-variants/components/variant-price-detail.tsx +87 -87
- package/src/app/routes/_authenticated/_product-variants/product-variants.graphql.ts +123 -123
- package/src/app/routes/_authenticated/_product-variants/product-variants.tsx +78 -78
- package/src/app/routes/_authenticated/_product-variants/product-variants_.$id.tsx +331 -331
- package/src/app/routes/_authenticated/_products/components/create-product-variants-dialog.tsx +228 -228
- package/src/app/routes/_authenticated/_products/components/create-product-variants.tsx +462 -462
- package/src/app/routes/_authenticated/_products/components/option-value-input.tsx +95 -95
- package/src/app/routes/_authenticated/_products/components/product-variants-table.tsx +87 -87
- package/src/app/routes/_authenticated/_products/products.graphql.ts +116 -116
- package/src/app/routes/_authenticated/_products/products.tsx +48 -48
- package/src/app/routes/_authenticated/_products/products_.$id.tsx +196 -196
- package/src/app/routes/_authenticated/_profile/profile.graphql.ts +23 -23
- package/src/app/routes/_authenticated/_profile/profile.tsx +122 -122
- package/src/app/routes/_authenticated/_promotions/components/promotion-actions-selector.tsx +107 -107
- package/src/app/routes/_authenticated/_promotions/components/promotion-conditions-selector.tsx +107 -107
- package/src/app/routes/_authenticated/_promotions/promotions.graphql.ts +96 -96
- package/src/app/routes/_authenticated/_promotions/promotions.tsx +61 -61
- package/src/app/routes/_authenticated/_promotions/promotions_.$id.tsx +235 -235
- package/src/app/routes/_authenticated/_roles/components/expandable-permissions.tsx +54 -54
- package/src/app/routes/_authenticated/_roles/components/permissions-grid.tsx +116 -116
- package/src/app/routes/_authenticated/_roles/roles.graphql.ts +67 -67
- package/src/app/routes/_authenticated/_roles/roles.tsx +96 -96
- package/src/app/routes/_authenticated/_roles/roles_.$id.tsx +142 -142
- package/src/app/routes/_authenticated/_sellers/sellers.graphql.ts +61 -61
- package/src/app/routes/_authenticated/_sellers/sellers.tsx +51 -51
- package/src/app/routes/_authenticated/_sellers/sellers_.$id.tsx +111 -111
- package/src/app/routes/_authenticated/_shipping-methods/components/fulfillment-handler-selector.tsx +56 -56
- package/src/app/routes/_authenticated/_shipping-methods/components/shipping-calculator-selector.tsx +101 -101
- package/src/app/routes/_authenticated/_shipping-methods/components/shipping-eligibility-checker-selector.tsx +101 -101
- package/src/app/routes/_authenticated/_shipping-methods/components/test-shipping-method-dialog.tsx +32 -32
- package/src/app/routes/_authenticated/_shipping-methods/shipping-methods.graphql.ts +83 -83
- package/src/app/routes/_authenticated/_shipping-methods/shipping-methods.tsx +55 -55
- package/src/app/routes/_authenticated/_shipping-methods/shipping-methods_.$id.tsx +171 -171
- package/src/app/routes/_authenticated/_stock-locations/stock-locations.graphql.ts +62 -62
- package/src/app/routes/_authenticated/_stock-locations/stock-locations.tsx +48 -48
- package/src/app/routes/_authenticated/_stock-locations/stock-locations_.$id.tsx +115 -115
- package/src/app/routes/_authenticated/_system/components/payload-dialog.tsx +34 -34
- package/src/app/routes/_authenticated/_system/healthchecks.tsx +93 -93
- package/src/app/routes/_authenticated/_system/job-queue.graphql.ts +43 -43
- package/src/app/routes/_authenticated/_system/job-queue.tsx +161 -161
- package/src/app/routes/_authenticated/_tax-categories/tax-categories.graphql.ts +63 -63
- package/src/app/routes/_authenticated/_tax-categories/tax-categories.tsx +65 -65
- package/src/app/routes/_authenticated/_tax-categories/tax-categories_.$id.tsx +115 -115
- package/src/app/routes/_authenticated/_tax-rates/tax-rates.graphql.ts +75 -75
- package/src/app/routes/_authenticated/_tax-rates/tax-rates.tsx +108 -108
- package/src/app/routes/_authenticated/_tax-rates/tax-rates_.$id.tsx +148 -148
- package/src/app/routes/_authenticated/_zones/components/zone-countries-sheet.tsx +31 -31
- package/src/app/routes/_authenticated/_zones/components/zone-countries-table.tsx +79 -79
- package/src/app/routes/_authenticated/_zones/zones.graphql.ts +96 -96
- package/src/app/routes/_authenticated/_zones/zones.tsx +57 -57
- package/src/app/routes/_authenticated/_zones/zones_.$id.tsx +103 -103
- package/src/app/routes/_authenticated/index.tsx +194 -194
- package/src/app/routes/_authenticated.tsx +25 -25
- package/src/app/routes/login.tsx +48 -48
- package/src/app/styles.css +82 -82
- package/src/app/tailwindcss-animate.css +275 -275
- package/src/i18n/locales/de.po +1579 -1579
- package/src/i18n/locales/en.po +1579 -1579
- package/src/lib/components/data-display/boolean.tsx +23 -23
- package/src/lib/components/data-display/date-time.tsx +13 -13
- package/src/lib/components/data-display/json.tsx +5 -5
- package/src/lib/components/data-display/money.tsx +15 -15
- package/src/lib/components/data-input/affixed-input.tsx +49 -49
- package/src/lib/components/data-input/customer-group-input.tsx +72 -72
- package/src/lib/components/data-input/datetime-input.tsx +149 -149
- package/src/lib/components/data-input/facet-value-input.tsx +68 -68
- package/src/lib/components/data-input/money-input.tsx +112 -112
- package/src/lib/components/data-input/richt-text-input.tsx +99 -99
- package/src/lib/components/data-table/data-table-column-header.tsx +73 -73
- package/src/lib/components/data-table/data-table-faceted-filter.tsx +172 -172
- package/src/lib/components/data-table/data-table-filter-dialog.tsx +73 -73
- package/src/lib/components/data-table/data-table-pagination.tsx +87 -87
- package/src/lib/components/data-table/data-table-view-options.tsx +53 -53
- package/src/lib/components/data-table/data-table.tsx +218 -218
- package/src/lib/components/layout/app-layout.tsx +42 -42
- package/src/lib/components/layout/app-sidebar.tsx +34 -34
- package/src/lib/components/layout/channel-switcher.tsx +87 -87
- package/src/lib/components/layout/content-language-selector.tsx +41 -41
- package/src/lib/components/layout/generated-breadcrumbs.tsx +82 -82
- package/src/lib/components/layout/language-dialog.tsx +104 -104
- package/src/lib/components/layout/nav-main.tsx +178 -178
- package/src/lib/components/layout/nav-projects.tsx +81 -81
- package/src/lib/components/layout/nav-user.tsx +176 -176
- package/src/lib/components/layout/prerelease-popup.tsx +38 -38
- package/src/lib/components/login/login-form.tsx +173 -173
- package/src/lib/components/shared/alerts.tsx +20 -20
- package/src/lib/components/shared/animated-number.tsx +49 -49
- package/src/lib/components/shared/asset-gallery.tsx +433 -433
- package/src/lib/components/shared/asset-picker-dialog.tsx +71 -71
- package/src/lib/components/shared/asset-preview-dialog.tsx +48 -48
- package/src/lib/components/shared/asset-preview.tsx +345 -345
- package/src/lib/components/shared/assigned-facet-values.tsx +68 -68
- package/src/lib/components/shared/channel-code-label.tsx +7 -7
- package/src/lib/components/shared/channel-selector.tsx +51 -51
- package/src/lib/components/shared/configurable-operation-arg-input.tsx +51 -51
- package/src/lib/components/shared/configurable-operation-input.tsx +133 -133
- package/src/lib/components/shared/confirmation-dialog.tsx +58 -58
- package/src/lib/components/shared/copyable-text.tsx +31 -31
- package/src/lib/components/shared/country-selector.tsx +105 -105
- package/src/lib/components/shared/currency-selector.tsx +33 -33
- package/src/lib/components/shared/custom-fields-form.tsx +86 -86
- package/src/lib/components/shared/customer-address-form.tsx +330 -330
- package/src/lib/components/shared/customer-group-chip.tsx +30 -30
- package/src/lib/components/shared/customer-group-selector.tsx +62 -62
- package/src/lib/components/shared/customer-selector.tsx +107 -107
- package/src/lib/components/shared/detail-page-button.tsx +22 -22
- package/src/lib/components/shared/entity-assets.tsx +340 -340
- package/src/lib/components/shared/error-page.tsx +31 -31
- package/src/lib/components/shared/facet-value-chip.tsx +44 -44
- package/src/lib/components/shared/facet-value-selector.tsx +306 -306
- package/src/lib/components/shared/focal-point-control.tsx +64 -64
- package/src/lib/components/shared/form-field-wrapper.tsx +37 -37
- package/src/lib/components/shared/history-timeline/history-entry.tsx +112 -112
- package/src/lib/components/shared/history-timeline/history-note-checkbox.tsx +28 -28
- package/src/lib/components/shared/history-timeline/history-note-editor.tsx +60 -60
- package/src/lib/components/shared/history-timeline/history-note-input.tsx +39 -39
- package/src/lib/components/shared/history-timeline/history-timeline.tsx +56 -56
- package/src/lib/components/shared/icon-mark.tsx +18 -18
- package/src/lib/components/shared/language-selector.tsx +48 -48
- package/src/lib/components/shared/logo-mark.tsx +24 -24
- package/src/lib/components/shared/multi-select.tsx +159 -159
- package/src/lib/components/shared/option-value-input.tsx +94 -94
- package/src/lib/components/shared/paginated-list-data-table.tsx +520 -520
- package/src/lib/components/shared/permission-guard.tsx +20 -20
- package/src/lib/components/shared/role-code-label.tsx +8 -8
- package/src/lib/components/shared/role-selector.tsx +56 -56
- package/src/lib/components/shared/seller-selector.tsx +107 -107
- package/src/lib/components/shared/tax-category-selector.tsx +65 -65
- package/src/lib/components/shared/translatable-form-field.tsx +74 -74
- package/src/lib/components/shared/vendure-image.tsx +159 -159
- package/src/lib/components/shared/zone-selector.tsx +66 -66
- package/src/lib/components/ui/accordion.tsx +59 -59
- package/src/lib/components/ui/alert-dialog.tsx +128 -128
- package/src/lib/components/ui/alert.tsx +60 -60
- package/src/lib/components/ui/avatar.tsx +38 -38
- package/src/lib/components/ui/badge.tsx +38 -38
- package/src/lib/components/ui/breadcrumb.tsx +102 -102
- package/src/lib/components/ui/button.tsx +51 -51
- package/src/lib/components/ui/calendar.tsx +69 -69
- package/src/lib/components/ui/card.tsx +47 -47
- package/src/lib/components/ui/checkbox.tsx +27 -27
- package/src/lib/components/ui/collapsible.tsx +33 -33
- package/src/lib/components/ui/command.tsx +133 -133
- package/src/lib/components/ui/dialog.tsx +116 -116
- package/src/lib/components/ui/dropdown-menu.tsx +220 -220
- package/src/lib/components/ui/form.tsx +141 -141
- package/src/lib/components/ui/hover-card.tsx +36 -36
- package/src/lib/components/ui/input.tsx +19 -19
- package/src/lib/components/ui/label.tsx +21 -21
- package/src/lib/components/ui/pagination.tsx +127 -127
- package/src/lib/components/ui/popover.tsx +40 -40
- package/src/lib/components/ui/scroll-area.tsx +50 -50
- package/src/lib/components/ui/select.tsx +161 -161
- package/src/lib/components/ui/separator.tsx +26 -26
- package/src/lib/components/ui/sheet.tsx +118 -118
- package/src/lib/components/ui/sidebar.tsx +696 -696
- package/src/lib/components/ui/skeleton.tsx +13 -13
- package/src/lib/components/ui/sonner.tsx +27 -27
- package/src/lib/components/ui/switch.tsx +26 -26
- package/src/lib/components/ui/table.tsx +82 -82
- package/src/lib/components/ui/tabs.tsx +48 -48
- package/src/lib/components/ui/textarea.tsx +18 -18
- package/src/lib/components/ui/tooltip.tsx +51 -51
- package/src/lib/constants.ts +326 -326
- package/src/lib/framework/component-registry/component-registry.tsx +70 -70
- package/src/lib/framework/component-registry/dynamic-component.tsx +58 -58
- package/src/lib/framework/dashboard-widget/base-widget.tsx +97 -97
- package/src/lib/framework/dashboard-widget/latest-orders-widget/index.tsx +96 -96
- package/src/lib/framework/dashboard-widget/latest-orders-widget/latest-orders-widget.graphql.ts +35 -35
- package/src/lib/framework/dashboard-widget/metrics-widget/chart.tsx +24 -24
- package/src/lib/framework/dashboard-widget/metrics-widget/index.tsx +82 -82
- package/src/lib/framework/dashboard-widget/metrics-widget/metrics-widget.graphql.ts +14 -14
- package/src/lib/framework/dashboard-widget/orders-summary/index.tsx +167 -167
- package/src/lib/framework/dashboard-widget/orders-summary/order-summary-widget.graphql.ts +14 -14
- package/src/lib/framework/dashboard-widget/types.ts +22 -22
- package/src/lib/framework/dashboard-widget/widget-extensions.tsx +19 -19
- package/src/lib/framework/defaults.ts +219 -219
- package/src/lib/framework/document-introspection/add-custom-fields.spec.ts +242 -242
- package/src/lib/framework/document-introspection/add-custom-fields.ts +246 -246
- package/src/lib/framework/document-introspection/get-document-structure.spec.ts +310 -310
- package/src/lib/framework/document-introspection/get-document-structure.ts +460 -460
- package/src/lib/framework/document-introspection/hooks.ts +10 -10
- package/src/lib/framework/extension-api/define-dashboard-extension.ts +66 -66
- package/src/lib/framework/extension-api/extension-api-types.ts +58 -58
- package/src/lib/framework/extension-api/use-dashboard-extensions.ts +26 -26
- package/src/lib/framework/form-engine/form-schema-tools.ts +98 -98
- package/src/lib/framework/form-engine/use-generated-form.tsx +116 -116
- package/src/lib/framework/layout-engine/layout-extensions.ts +30 -30
- package/src/lib/framework/layout-engine/location-wrapper.tsx +96 -96
- package/src/lib/framework/layout-engine/page-layout.tsx +272 -272
- package/src/lib/framework/nav-menu/nav-menu-extensions.ts +66 -66
- package/src/lib/framework/page/detail-page-route-loader.tsx +48 -48
- package/src/lib/framework/page/detail-page.tsx +131 -131
- package/src/lib/framework/page/list-page.tsx +166 -166
- package/src/lib/framework/page/page-api.ts +9 -9
- package/src/lib/framework/page/page-types.ts +51 -51
- package/src/lib/framework/page/use-detail-page.ts +217 -217
- package/src/lib/framework/page/use-extended-router.tsx +69 -69
- package/src/lib/framework/registry/global-registry.ts +46 -46
- package/src/lib/framework/registry/registry-types.ts +15 -15
- package/src/lib/graphql/api.ts +61 -61
- package/src/lib/graphql/fragments.tsx +54 -54
- package/src/lib/graphql/graphql-env.d.ts +499 -499
- package/src/lib/graphql/graphql.ts +15 -15
- package/src/lib/hooks/use-auth.tsx +11 -11
- package/src/lib/hooks/use-channel.ts +12 -12
- package/src/lib/hooks/use-custom-field-config.ts +10 -10
- package/src/lib/hooks/use-grouped-permissions.ts +54 -54
- package/src/lib/hooks/use-local-format.ts +119 -119
- package/src/lib/hooks/use-mobile.ts +19 -19
- package/src/lib/hooks/use-page.tsx +10 -10
- package/src/lib/hooks/use-permissions.ts +22 -22
- package/src/lib/hooks/use-server-config.ts +4 -4
- package/src/lib/hooks/use-theme.ts +10 -10
- package/src/lib/hooks/use-user-settings.tsx +12 -12
- package/src/lib/index.ts +149 -149
- package/src/lib/lib/trans.tsx +16 -16
- package/src/lib/lib/utils.ts +60 -60
- package/src/lib/providers/auth.tsx +152 -152
- package/src/lib/providers/channel-provider.tsx +121 -121
- package/src/lib/providers/i18n-provider.tsx +28 -28
- package/src/lib/providers/server-config.tsx +279 -279
- package/src/lib/providers/theme-provider.tsx +54 -54
- package/src/lib/providers/user-settings.tsx +89 -89
- package/src/lib/virtual.d.ts +12 -12
- package/vite/config-loader.ts +181 -181
- package/vite/constants.ts +280 -280
- package/vite/index.ts +1 -1
- package/vite/schema-generator.ts +40 -40
- package/vite/ui-config.ts +60 -60
- package/vite/vite-plugin-admin-api-schema.ts +141 -141
- package/vite/vite-plugin-config-loader.ts +64 -64
- package/vite/vite-plugin-config.ts +42 -42
- package/vite/vite-plugin-dashboard-metadata.ts +58 -58
- package/vite/vite-plugin-gql-tada.ts +62 -62
- package/vite/vite-plugin-theme.ts +195 -195
- package/vite/vite-plugin-ui-config.ts +60 -60
- package/vite/vite-plugin-vendure-dashboard.ts +118 -118
- package/dist/plugin/.vendure-dashboard-temp/dev-config.js +0 -227
- package/dist/plugin/.vendure-dashboard-temp/dev-config.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/api-extensions.js +0 -33
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/api-extensions.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/mv.resolver.js +0 -69
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/mv.resolver.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-process.js +0 -110
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-process.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-seller-strategy.js +0 -134
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-seller-strategy.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-payment-handler.js +0 -86
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-payment-handler.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-eligibility-checker.js +0 -49
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-eligibility-checker.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-line-assignment-strategy.js +0 -57
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-line-assignment-strategy.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/constants.js +0 -20
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/constants.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/multivendor.plugin.js +0 -151
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/multivendor.plugin.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/payment/mv-connect-sdk.js +0 -47
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/payment/mv-connect-sdk.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/service/mv.service.js +0 -222
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/service/mv.service.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/types.js +0 -4
- package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/types.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/package.json +0 -3
- package/dist/plugin/.vendure-dashboard-temp/schema.graphql +0 -6378
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/api-extensions.js +0 -103
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/api-extensions.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-entity.resolver.js +0 -105
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-entity.resolver.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-admin.resolver.js +0 -183
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-admin.resolver.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-entity.resolver.js +0 -113
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-entity.resolver.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-shop.resolver.js +0 -112
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-shop.resolver.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/entities/product-review.entity.js +0 -111
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/entities/product-review.entity.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-admin-types.js +0 -616
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-admin-types.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-shop-types.js +0 -563
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-shop-types.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/reviews-plugin.js +0 -135
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/reviews-plugin.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/types.js +0 -4
- package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/types.js.map +0 -1
- package/dist/plugin/.vendure-dashboard-temp/tsconfig.json +0 -10
- package/dist/plugin/vite-plugin-set-root.d.ts +0 -4
- package/dist/plugin/vite-plugin-set-root.js +0 -15
- 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
|
+
}
|