@vendure/dashboard 3.2.0 → 3.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (381) hide show
  1. package/LICENSE.md +42 -0
  2. package/README.md +12 -12
  3. package/index.html +15 -15
  4. package/lingui.config.js +12 -12
  5. package/package.json +11 -2
  6. package/src/app/app-providers.tsx +30 -30
  7. package/src/app/main.tsx +97 -97
  8. package/src/app/routes/__root.tsx +24 -24
  9. package/src/app/routes/_authenticated/_administrators/administrators.graphql.ts +79 -79
  10. package/src/app/routes/_authenticated/_administrators/administrators.tsx +86 -86
  11. package/src/app/routes/_authenticated/_administrators/administrators_.$id.tsx +155 -155
  12. package/src/app/routes/_authenticated/_administrators/components/role-permissions-display.tsx +133 -133
  13. package/src/app/routes/_authenticated/_assets/assets.tsx +19 -19
  14. package/src/app/routes/_authenticated/_channels/channels.graphql.ts +93 -93
  15. package/src/app/routes/_authenticated/_channels/channels.tsx +60 -60
  16. package/src/app/routes/_authenticated/_channels/channels_.$id.tsx +248 -248
  17. package/src/app/routes/_authenticated/_collections/collections.graphql.ts +133 -133
  18. package/src/app/routes/_authenticated/_collections/collections.tsx +195 -195
  19. package/src/app/routes/_authenticated/_collections/collections_.$id.tsx +224 -224
  20. package/src/app/routes/_authenticated/_collections/components/collection-contents-preview-table.tsx +127 -127
  21. package/src/app/routes/_authenticated/_collections/components/collection-contents-sheet.tsx +46 -46
  22. package/src/app/routes/_authenticated/_collections/components/collection-contents-table.tsx +82 -82
  23. package/src/app/routes/_authenticated/_collections/components/collection-filters-selector.tsx +91 -91
  24. package/src/app/routes/_authenticated/_countries/countries.graphql.ts +69 -69
  25. package/src/app/routes/_authenticated/_countries/countries.tsx +67 -67
  26. package/src/app/routes/_authenticated/_countries/countries_.$id.tsx +122 -122
  27. package/src/app/routes/_authenticated/_customer-groups/components/customer-group-members-sheet.tsx +44 -44
  28. package/src/app/routes/_authenticated/_customer-groups/components/customer-group-members-table.tsx +129 -129
  29. package/src/app/routes/_authenticated/_customer-groups/customer-groups.graphql.ts +71 -71
  30. package/src/app/routes/_authenticated/_customer-groups/customer-groups.tsx +68 -68
  31. package/src/app/routes/_authenticated/_customer-groups/customer-groups_.$id.tsx +111 -111
  32. package/src/app/routes/_authenticated/_customers/components/customer-address-card.tsx +155 -155
  33. package/src/app/routes/_authenticated/_customers/components/customer-address-form.tsx +344 -344
  34. package/src/app/routes/_authenticated/_customers/components/customer-group-controls.tsx +4 -4
  35. package/src/app/routes/_authenticated/_customers/components/customer-history/customer-history-container.tsx +78 -78
  36. package/src/app/routes/_authenticated/_customers/components/customer-history/customer-history.tsx +77 -77
  37. package/src/app/routes/_authenticated/_customers/components/customer-history/index.ts +3 -3
  38. package/src/app/routes/_authenticated/_customers/components/customer-history/use-customer-history.ts +169 -169
  39. package/src/app/routes/_authenticated/_customers/components/customer-order-table.tsx +88 -88
  40. package/src/app/routes/_authenticated/_customers/components/customer-status-badge.tsx +33 -33
  41. package/src/app/routes/_authenticated/_customers/customers.graphql.ts +204 -204
  42. package/src/app/routes/_authenticated/_customers/customers.tsx +82 -82
  43. package/src/app/routes/_authenticated/_customers/customers_.$id.tsx +274 -274
  44. package/src/app/routes/_authenticated/_facets/components/edit-facet-value.tsx +129 -129
  45. package/src/app/routes/_authenticated/_facets/components/facet-values-sheet.tsx +46 -46
  46. package/src/app/routes/_authenticated/_facets/components/facet-values-table.tsx +97 -97
  47. package/src/app/routes/_authenticated/_facets/facets.graphql.ts +104 -104
  48. package/src/app/routes/_authenticated/_facets/facets.tsx +97 -97
  49. package/src/app/routes/_authenticated/_facets/facets_.$id.tsx +139 -139
  50. package/src/app/routes/_authenticated/_global-settings/global-settings.graphql.ts +28 -28
  51. package/src/app/routes/_authenticated/_global-settings/global-settings.tsx +161 -161
  52. package/src/app/routes/_authenticated/_orders/components/order-address.tsx +58 -58
  53. package/src/app/routes/_authenticated/_orders/components/order-history/index.ts +3 -3
  54. package/src/app/routes/_authenticated/_orders/components/order-history/order-history-container.tsx +72 -72
  55. package/src/app/routes/_authenticated/_orders/components/order-history/order-history.tsx +96 -96
  56. package/src/app/routes/_authenticated/_orders/components/order-history/use-order-history.ts +171 -171
  57. package/src/app/routes/_authenticated/_orders/components/order-table.tsx +169 -169
  58. package/src/app/routes/_authenticated/_orders/components/order-tax-summary.tsx +38 -38
  59. package/src/app/routes/_authenticated/_orders/components/payment-details.tsx +61 -61
  60. package/src/app/routes/_authenticated/_orders/orders.graphql.ts +325 -325
  61. package/src/app/routes/_authenticated/_orders/orders.tsx +120 -120
  62. package/src/app/routes/_authenticated/_orders/orders_.$id.tsx +133 -133
  63. package/src/app/routes/_authenticated/_payment-methods/components/payment-eligibility-checker-selector.tsx +104 -104
  64. package/src/app/routes/_authenticated/_payment-methods/components/payment-handler-selector.tsx +100 -100
  65. package/src/app/routes/_authenticated/_payment-methods/payment-methods.graphql.ts +83 -83
  66. package/src/app/routes/_authenticated/_payment-methods/payment-methods.tsx +64 -64
  67. package/src/app/routes/_authenticated/_payment-methods/payment-methods_.$id.tsx +183 -183
  68. package/src/app/routes/_authenticated/_product-variants/components/variant-price-detail.tsx +87 -87
  69. package/src/app/routes/_authenticated/_product-variants/product-variants.graphql.ts +123 -123
  70. package/src/app/routes/_authenticated/_product-variants/product-variants.tsx +78 -78
  71. package/src/app/routes/_authenticated/_product-variants/product-variants_.$id.tsx +331 -331
  72. package/src/app/routes/_authenticated/_products/components/create-product-variants-dialog.tsx +228 -228
  73. package/src/app/routes/_authenticated/_products/components/create-product-variants.tsx +462 -462
  74. package/src/app/routes/_authenticated/_products/components/option-value-input.tsx +95 -95
  75. package/src/app/routes/_authenticated/_products/components/product-variants-table.tsx +87 -87
  76. package/src/app/routes/_authenticated/_products/products.graphql.ts +116 -116
  77. package/src/app/routes/_authenticated/_products/products.tsx +48 -48
  78. package/src/app/routes/_authenticated/_products/products_.$id.tsx +196 -196
  79. package/src/app/routes/_authenticated/_profile/profile.graphql.ts +23 -23
  80. package/src/app/routes/_authenticated/_profile/profile.tsx +122 -122
  81. package/src/app/routes/_authenticated/_promotions/components/promotion-actions-selector.tsx +107 -107
  82. package/src/app/routes/_authenticated/_promotions/components/promotion-conditions-selector.tsx +107 -107
  83. package/src/app/routes/_authenticated/_promotions/promotions.graphql.ts +96 -96
  84. package/src/app/routes/_authenticated/_promotions/promotions.tsx +61 -61
  85. package/src/app/routes/_authenticated/_promotions/promotions_.$id.tsx +235 -235
  86. package/src/app/routes/_authenticated/_roles/components/expandable-permissions.tsx +54 -54
  87. package/src/app/routes/_authenticated/_roles/components/permissions-grid.tsx +116 -116
  88. package/src/app/routes/_authenticated/_roles/roles.graphql.ts +67 -67
  89. package/src/app/routes/_authenticated/_roles/roles.tsx +96 -96
  90. package/src/app/routes/_authenticated/_roles/roles_.$id.tsx +142 -142
  91. package/src/app/routes/_authenticated/_sellers/sellers.graphql.ts +61 -61
  92. package/src/app/routes/_authenticated/_sellers/sellers.tsx +51 -51
  93. package/src/app/routes/_authenticated/_sellers/sellers_.$id.tsx +111 -111
  94. package/src/app/routes/_authenticated/_shipping-methods/components/fulfillment-handler-selector.tsx +56 -56
  95. package/src/app/routes/_authenticated/_shipping-methods/components/shipping-calculator-selector.tsx +101 -101
  96. package/src/app/routes/_authenticated/_shipping-methods/components/shipping-eligibility-checker-selector.tsx +101 -101
  97. package/src/app/routes/_authenticated/_shipping-methods/components/test-shipping-method-dialog.tsx +32 -32
  98. package/src/app/routes/_authenticated/_shipping-methods/shipping-methods.graphql.ts +83 -83
  99. package/src/app/routes/_authenticated/_shipping-methods/shipping-methods.tsx +55 -55
  100. package/src/app/routes/_authenticated/_shipping-methods/shipping-methods_.$id.tsx +171 -171
  101. package/src/app/routes/_authenticated/_stock-locations/stock-locations.graphql.ts +62 -62
  102. package/src/app/routes/_authenticated/_stock-locations/stock-locations.tsx +48 -48
  103. package/src/app/routes/_authenticated/_stock-locations/stock-locations_.$id.tsx +115 -115
  104. package/src/app/routes/_authenticated/_system/components/payload-dialog.tsx +34 -34
  105. package/src/app/routes/_authenticated/_system/healthchecks.tsx +93 -93
  106. package/src/app/routes/_authenticated/_system/job-queue.graphql.ts +43 -43
  107. package/src/app/routes/_authenticated/_system/job-queue.tsx +161 -161
  108. package/src/app/routes/_authenticated/_tax-categories/tax-categories.graphql.ts +63 -63
  109. package/src/app/routes/_authenticated/_tax-categories/tax-categories.tsx +65 -65
  110. package/src/app/routes/_authenticated/_tax-categories/tax-categories_.$id.tsx +115 -115
  111. package/src/app/routes/_authenticated/_tax-rates/tax-rates.graphql.ts +75 -75
  112. package/src/app/routes/_authenticated/_tax-rates/tax-rates.tsx +108 -108
  113. package/src/app/routes/_authenticated/_tax-rates/tax-rates_.$id.tsx +148 -148
  114. package/src/app/routes/_authenticated/_zones/components/zone-countries-sheet.tsx +31 -31
  115. package/src/app/routes/_authenticated/_zones/components/zone-countries-table.tsx +79 -79
  116. package/src/app/routes/_authenticated/_zones/zones.graphql.ts +96 -96
  117. package/src/app/routes/_authenticated/_zones/zones.tsx +57 -57
  118. package/src/app/routes/_authenticated/_zones/zones_.$id.tsx +103 -103
  119. package/src/app/routes/_authenticated/index.tsx +194 -194
  120. package/src/app/routes/_authenticated.tsx +25 -25
  121. package/src/app/routes/login.tsx +48 -48
  122. package/src/app/styles.css +82 -82
  123. package/src/app/tailwindcss-animate.css +275 -275
  124. package/src/i18n/locales/de.po +1579 -1579
  125. package/src/i18n/locales/en.po +1579 -1579
  126. package/src/lib/components/data-display/boolean.tsx +23 -23
  127. package/src/lib/components/data-display/date-time.tsx +13 -13
  128. package/src/lib/components/data-display/json.tsx +5 -5
  129. package/src/lib/components/data-display/money.tsx +15 -15
  130. package/src/lib/components/data-input/affixed-input.tsx +49 -49
  131. package/src/lib/components/data-input/customer-group-input.tsx +72 -72
  132. package/src/lib/components/data-input/datetime-input.tsx +149 -149
  133. package/src/lib/components/data-input/facet-value-input.tsx +68 -68
  134. package/src/lib/components/data-input/money-input.tsx +112 -112
  135. package/src/lib/components/data-input/richt-text-input.tsx +99 -99
  136. package/src/lib/components/data-table/data-table-column-header.tsx +73 -73
  137. package/src/lib/components/data-table/data-table-faceted-filter.tsx +172 -172
  138. package/src/lib/components/data-table/data-table-filter-dialog.tsx +73 -73
  139. package/src/lib/components/data-table/data-table-pagination.tsx +87 -87
  140. package/src/lib/components/data-table/data-table-view-options.tsx +53 -53
  141. package/src/lib/components/data-table/data-table.tsx +218 -218
  142. package/src/lib/components/layout/app-layout.tsx +42 -42
  143. package/src/lib/components/layout/app-sidebar.tsx +34 -34
  144. package/src/lib/components/layout/channel-switcher.tsx +87 -87
  145. package/src/lib/components/layout/content-language-selector.tsx +41 -41
  146. package/src/lib/components/layout/generated-breadcrumbs.tsx +82 -82
  147. package/src/lib/components/layout/language-dialog.tsx +104 -104
  148. package/src/lib/components/layout/nav-main.tsx +178 -178
  149. package/src/lib/components/layout/nav-projects.tsx +81 -81
  150. package/src/lib/components/layout/nav-user.tsx +176 -176
  151. package/src/lib/components/layout/prerelease-popup.tsx +38 -38
  152. package/src/lib/components/login/login-form.tsx +173 -173
  153. package/src/lib/components/shared/alerts.tsx +20 -20
  154. package/src/lib/components/shared/animated-number.tsx +49 -49
  155. package/src/lib/components/shared/asset-gallery.tsx +433 -433
  156. package/src/lib/components/shared/asset-picker-dialog.tsx +71 -71
  157. package/src/lib/components/shared/asset-preview-dialog.tsx +48 -48
  158. package/src/lib/components/shared/asset-preview.tsx +345 -345
  159. package/src/lib/components/shared/assigned-facet-values.tsx +68 -68
  160. package/src/lib/components/shared/channel-code-label.tsx +7 -7
  161. package/src/lib/components/shared/channel-selector.tsx +51 -51
  162. package/src/lib/components/shared/configurable-operation-arg-input.tsx +51 -51
  163. package/src/lib/components/shared/configurable-operation-input.tsx +133 -133
  164. package/src/lib/components/shared/confirmation-dialog.tsx +58 -58
  165. package/src/lib/components/shared/copyable-text.tsx +31 -31
  166. package/src/lib/components/shared/country-selector.tsx +105 -105
  167. package/src/lib/components/shared/currency-selector.tsx +33 -33
  168. package/src/lib/components/shared/custom-fields-form.tsx +86 -86
  169. package/src/lib/components/shared/customer-address-form.tsx +330 -330
  170. package/src/lib/components/shared/customer-group-chip.tsx +30 -30
  171. package/src/lib/components/shared/customer-group-selector.tsx +62 -62
  172. package/src/lib/components/shared/customer-selector.tsx +107 -107
  173. package/src/lib/components/shared/detail-page-button.tsx +22 -22
  174. package/src/lib/components/shared/entity-assets.tsx +340 -340
  175. package/src/lib/components/shared/error-page.tsx +31 -31
  176. package/src/lib/components/shared/facet-value-chip.tsx +44 -44
  177. package/src/lib/components/shared/facet-value-selector.tsx +306 -306
  178. package/src/lib/components/shared/focal-point-control.tsx +64 -64
  179. package/src/lib/components/shared/form-field-wrapper.tsx +37 -37
  180. package/src/lib/components/shared/history-timeline/history-entry.tsx +112 -112
  181. package/src/lib/components/shared/history-timeline/history-note-checkbox.tsx +28 -28
  182. package/src/lib/components/shared/history-timeline/history-note-editor.tsx +60 -60
  183. package/src/lib/components/shared/history-timeline/history-note-input.tsx +39 -39
  184. package/src/lib/components/shared/history-timeline/history-timeline.tsx +56 -56
  185. package/src/lib/components/shared/icon-mark.tsx +18 -18
  186. package/src/lib/components/shared/language-selector.tsx +48 -48
  187. package/src/lib/components/shared/logo-mark.tsx +24 -24
  188. package/src/lib/components/shared/multi-select.tsx +159 -159
  189. package/src/lib/components/shared/option-value-input.tsx +94 -94
  190. package/src/lib/components/shared/paginated-list-data-table.tsx +520 -520
  191. package/src/lib/components/shared/permission-guard.tsx +20 -20
  192. package/src/lib/components/shared/role-code-label.tsx +8 -8
  193. package/src/lib/components/shared/role-selector.tsx +56 -56
  194. package/src/lib/components/shared/seller-selector.tsx +107 -107
  195. package/src/lib/components/shared/tax-category-selector.tsx +65 -65
  196. package/src/lib/components/shared/translatable-form-field.tsx +74 -74
  197. package/src/lib/components/shared/vendure-image.tsx +159 -159
  198. package/src/lib/components/shared/zone-selector.tsx +66 -66
  199. package/src/lib/components/ui/accordion.tsx +59 -59
  200. package/src/lib/components/ui/alert-dialog.tsx +128 -128
  201. package/src/lib/components/ui/alert.tsx +60 -60
  202. package/src/lib/components/ui/avatar.tsx +38 -38
  203. package/src/lib/components/ui/badge.tsx +38 -38
  204. package/src/lib/components/ui/breadcrumb.tsx +102 -102
  205. package/src/lib/components/ui/button.tsx +51 -51
  206. package/src/lib/components/ui/calendar.tsx +69 -69
  207. package/src/lib/components/ui/card.tsx +47 -47
  208. package/src/lib/components/ui/checkbox.tsx +27 -27
  209. package/src/lib/components/ui/collapsible.tsx +33 -33
  210. package/src/lib/components/ui/command.tsx +133 -133
  211. package/src/lib/components/ui/dialog.tsx +116 -116
  212. package/src/lib/components/ui/dropdown-menu.tsx +220 -220
  213. package/src/lib/components/ui/form.tsx +141 -141
  214. package/src/lib/components/ui/hover-card.tsx +36 -36
  215. package/src/lib/components/ui/input.tsx +19 -19
  216. package/src/lib/components/ui/label.tsx +21 -21
  217. package/src/lib/components/ui/pagination.tsx +127 -127
  218. package/src/lib/components/ui/popover.tsx +40 -40
  219. package/src/lib/components/ui/scroll-area.tsx +50 -50
  220. package/src/lib/components/ui/select.tsx +161 -161
  221. package/src/lib/components/ui/separator.tsx +26 -26
  222. package/src/lib/components/ui/sheet.tsx +118 -118
  223. package/src/lib/components/ui/sidebar.tsx +696 -696
  224. package/src/lib/components/ui/skeleton.tsx +13 -13
  225. package/src/lib/components/ui/sonner.tsx +27 -27
  226. package/src/lib/components/ui/switch.tsx +26 -26
  227. package/src/lib/components/ui/table.tsx +82 -82
  228. package/src/lib/components/ui/tabs.tsx +48 -48
  229. package/src/lib/components/ui/textarea.tsx +18 -18
  230. package/src/lib/components/ui/tooltip.tsx +51 -51
  231. package/src/lib/constants.ts +326 -326
  232. package/src/lib/framework/component-registry/component-registry.tsx +70 -70
  233. package/src/lib/framework/component-registry/dynamic-component.tsx +58 -58
  234. package/src/lib/framework/dashboard-widget/base-widget.tsx +97 -97
  235. package/src/lib/framework/dashboard-widget/latest-orders-widget/index.tsx +96 -96
  236. package/src/lib/framework/dashboard-widget/latest-orders-widget/latest-orders-widget.graphql.ts +35 -35
  237. package/src/lib/framework/dashboard-widget/metrics-widget/chart.tsx +24 -24
  238. package/src/lib/framework/dashboard-widget/metrics-widget/index.tsx +82 -82
  239. package/src/lib/framework/dashboard-widget/metrics-widget/metrics-widget.graphql.ts +14 -14
  240. package/src/lib/framework/dashboard-widget/orders-summary/index.tsx +167 -167
  241. package/src/lib/framework/dashboard-widget/orders-summary/order-summary-widget.graphql.ts +14 -14
  242. package/src/lib/framework/dashboard-widget/types.ts +22 -22
  243. package/src/lib/framework/dashboard-widget/widget-extensions.tsx +19 -19
  244. package/src/lib/framework/defaults.ts +219 -219
  245. package/src/lib/framework/document-introspection/add-custom-fields.spec.ts +242 -242
  246. package/src/lib/framework/document-introspection/add-custom-fields.ts +246 -246
  247. package/src/lib/framework/document-introspection/get-document-structure.spec.ts +310 -310
  248. package/src/lib/framework/document-introspection/get-document-structure.ts +460 -460
  249. package/src/lib/framework/document-introspection/hooks.ts +10 -10
  250. package/src/lib/framework/extension-api/define-dashboard-extension.ts +66 -66
  251. package/src/lib/framework/extension-api/extension-api-types.ts +58 -58
  252. package/src/lib/framework/extension-api/use-dashboard-extensions.ts +26 -26
  253. package/src/lib/framework/form-engine/form-schema-tools.ts +98 -98
  254. package/src/lib/framework/form-engine/use-generated-form.tsx +116 -116
  255. package/src/lib/framework/layout-engine/layout-extensions.ts +30 -30
  256. package/src/lib/framework/layout-engine/location-wrapper.tsx +96 -96
  257. package/src/lib/framework/layout-engine/page-layout.tsx +272 -272
  258. package/src/lib/framework/nav-menu/nav-menu-extensions.ts +66 -66
  259. package/src/lib/framework/page/detail-page-route-loader.tsx +48 -48
  260. package/src/lib/framework/page/detail-page.tsx +131 -131
  261. package/src/lib/framework/page/list-page.tsx +166 -166
  262. package/src/lib/framework/page/page-api.ts +9 -9
  263. package/src/lib/framework/page/page-types.ts +51 -51
  264. package/src/lib/framework/page/use-detail-page.ts +217 -217
  265. package/src/lib/framework/page/use-extended-router.tsx +69 -69
  266. package/src/lib/framework/registry/global-registry.ts +46 -46
  267. package/src/lib/framework/registry/registry-types.ts +15 -15
  268. package/src/lib/graphql/api.ts +61 -61
  269. package/src/lib/graphql/fragments.tsx +54 -54
  270. package/src/lib/graphql/graphql-env.d.ts +499 -499
  271. package/src/lib/graphql/graphql.ts +15 -15
  272. package/src/lib/hooks/use-auth.tsx +11 -11
  273. package/src/lib/hooks/use-channel.ts +12 -12
  274. package/src/lib/hooks/use-custom-field-config.ts +10 -10
  275. package/src/lib/hooks/use-grouped-permissions.ts +54 -54
  276. package/src/lib/hooks/use-local-format.ts +119 -119
  277. package/src/lib/hooks/use-mobile.ts +19 -19
  278. package/src/lib/hooks/use-page.tsx +10 -10
  279. package/src/lib/hooks/use-permissions.ts +22 -22
  280. package/src/lib/hooks/use-server-config.ts +4 -4
  281. package/src/lib/hooks/use-theme.ts +10 -10
  282. package/src/lib/hooks/use-user-settings.tsx +12 -12
  283. package/src/lib/index.ts +149 -149
  284. package/src/lib/lib/trans.tsx +16 -16
  285. package/src/lib/lib/utils.ts +60 -60
  286. package/src/lib/providers/auth.tsx +152 -152
  287. package/src/lib/providers/channel-provider.tsx +121 -121
  288. package/src/lib/providers/i18n-provider.tsx +28 -28
  289. package/src/lib/providers/server-config.tsx +279 -279
  290. package/src/lib/providers/theme-provider.tsx +54 -54
  291. package/src/lib/providers/user-settings.tsx +89 -89
  292. package/src/lib/virtual.d.ts +12 -12
  293. package/vite/config-loader.ts +181 -181
  294. package/vite/constants.ts +280 -280
  295. package/vite/index.ts +1 -1
  296. package/vite/schema-generator.ts +40 -40
  297. package/vite/ui-config.ts +60 -60
  298. package/vite/vite-plugin-admin-api-schema.ts +141 -141
  299. package/vite/vite-plugin-config-loader.ts +64 -64
  300. package/vite/vite-plugin-config.ts +42 -42
  301. package/vite/vite-plugin-dashboard-metadata.ts +58 -58
  302. package/vite/vite-plugin-gql-tada.ts +62 -62
  303. package/vite/vite-plugin-theme.ts +195 -195
  304. package/vite/vite-plugin-ui-config.ts +60 -60
  305. package/vite/vite-plugin-vendure-dashboard.ts +118 -118
  306. package/dist/plugin/.vendure-dashboard-temp/dev-config.js +0 -227
  307. package/dist/plugin/.vendure-dashboard-temp/dev-config.js.map +0 -1
  308. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/api-extensions.js +0 -33
  309. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/api-extensions.js.map +0 -1
  310. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/mv.resolver.js +0 -69
  311. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/api/mv.resolver.js.map +0 -1
  312. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-process.js +0 -110
  313. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-process.js.map +0 -1
  314. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-seller-strategy.js +0 -134
  315. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-order-seller-strategy.js.map +0 -1
  316. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-payment-handler.js +0 -86
  317. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-payment-handler.js.map +0 -1
  318. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-eligibility-checker.js +0 -49
  319. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-eligibility-checker.js.map +0 -1
  320. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-line-assignment-strategy.js +0 -57
  321. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/config/mv-shipping-line-assignment-strategy.js.map +0 -1
  322. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/constants.js +0 -20
  323. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/constants.js.map +0 -1
  324. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/multivendor.plugin.js +0 -151
  325. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/multivendor.plugin.js.map +0 -1
  326. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/payment/mv-connect-sdk.js +0 -47
  327. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/payment/mv-connect-sdk.js.map +0 -1
  328. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/service/mv.service.js +0 -222
  329. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/service/mv.service.js.map +0 -1
  330. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/types.js +0 -4
  331. package/dist/plugin/.vendure-dashboard-temp/example-plugins/multivendor-plugin/types.js.map +0 -1
  332. package/dist/plugin/.vendure-dashboard-temp/package.json +0 -3
  333. package/dist/plugin/.vendure-dashboard-temp/schema.graphql +0 -6378
  334. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/api-extensions.js +0 -103
  335. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/api-extensions.js.map +0 -1
  336. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-entity.resolver.js +0 -105
  337. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-entity.resolver.js.map +0 -1
  338. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-admin.resolver.js +0 -183
  339. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-admin.resolver.js.map +0 -1
  340. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-entity.resolver.js +0 -113
  341. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-entity.resolver.js.map +0 -1
  342. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-shop.resolver.js +0 -112
  343. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/api/product-review-shop.resolver.js.map +0 -1
  344. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/entities/product-review.entity.js +0 -111
  345. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/entities/product-review.entity.js.map +0 -1
  346. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-admin-types.js +0 -616
  347. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-admin-types.js.map +0 -1
  348. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-shop-types.js +0 -563
  349. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/generated-shop-types.js.map +0 -1
  350. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/reviews-plugin.js +0 -135
  351. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/reviews-plugin.js.map +0 -1
  352. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/types.js +0 -4
  353. package/dist/plugin/.vendure-dashboard-temp/test-plugins/reviews/types.js.map +0 -1
  354. package/dist/plugin/.vendure-dashboard-temp/tsconfig.json +0 -10
  355. package/dist/plugin/config-loader.d.ts +0 -27
  356. package/dist/plugin/config-loader.js +0 -141
  357. package/dist/plugin/constants.d.ts +0 -5
  358. package/dist/plugin/constants.js +0 -277
  359. package/dist/plugin/index.d.ts +0 -1
  360. package/dist/plugin/index.js +0 -1
  361. package/dist/plugin/schema-generator.d.ts +0 -5
  362. package/dist/plugin/schema-generator.js +0 -24
  363. package/dist/plugin/ui-config.d.ts +0 -3
  364. package/dist/plugin/ui-config.js +0 -35
  365. package/dist/plugin/vite-plugin-admin-api-schema.d.ts +0 -24
  366. package/dist/plugin/vite-plugin-admin-api-schema.js +0 -82
  367. package/dist/plugin/vite-plugin-config-loader.d.ts +0 -17
  368. package/dist/plugin/vite-plugin-config-loader.js +0 -56
  369. package/dist/plugin/vite-plugin-config.d.ts +0 -4
  370. package/dist/plugin/vite-plugin-config.js +0 -38
  371. package/dist/plugin/vite-plugin-dashboard-metadata.d.ts +0 -9
  372. package/dist/plugin/vite-plugin-dashboard-metadata.js +0 -52
  373. package/dist/plugin/vite-plugin-gql-tada.d.ts +0 -6
  374. package/dist/plugin/vite-plugin-gql-tada.js +0 -51
  375. package/dist/plugin/vite-plugin-set-root.d.ts +0 -4
  376. package/dist/plugin/vite-plugin-set-root.js +0 -15
  377. package/dist/plugin/vite-plugin-ui-config.d.ts +0 -15
  378. package/dist/plugin/vite-plugin-ui-config.js +0 -43
  379. package/dist/plugin/vite-plugin-vendure-dashboard.d.ts +0 -35
  380. package/dist/plugin/vite-plugin-vendure-dashboard.js +0 -81
  381. package/src/app/routeTree.gen.ts +0 -1372
@@ -1,696 +1,696 @@
1
- 'use client';
2
-
3
- import { MouseEvent } from 'react';
4
- import * as React from 'react';
5
- import { Slot } from '@radix-ui/react-slot';
6
- import { VariantProps, cva } from 'class-variance-authority';
7
- import { PanelLeftIcon } from 'lucide-react';
8
-
9
- import { useIsMobile } from '@/hooks/use-mobile.js';
10
- import { cn } from '@/lib/utils.js';
11
- import { Button } from '@/components/ui/button.js';
12
- import { Input } from '@/components/ui/input.js';
13
- import { Separator } from '@/components/ui/separator.js';
14
- import { Sheet, SheetContent, SheetDescription, SheetHeader, SheetTitle } from '@/components/ui/sheet.js';
15
- import { Skeleton } from '@/components/ui/skeleton.js';
16
- import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip.js';
17
-
18
- const SIDEBAR_COOKIE_NAME = 'sidebar_state';
19
- const SIDEBAR_COOKIE_MAX_AGE = 60 * 60 * 24 * 7;
20
- const SIDEBAR_WIDTH = '16rem';
21
- const SIDEBAR_WIDTH_MOBILE = '18rem';
22
- const SIDEBAR_WIDTH_ICON = '3rem';
23
- const SIDEBAR_KEYBOARD_SHORTCUT = 'b';
24
-
25
- type SidebarContext = {
26
- state: 'expanded' | 'collapsed';
27
- open: boolean;
28
- setOpen: (open: boolean) => void;
29
- openMobile: boolean;
30
- setOpenMobile: (open: boolean) => void;
31
- isMobile: boolean;
32
- toggleSidebar: () => void;
33
- };
34
-
35
- const SidebarContext = React.createContext<SidebarContext | null>(null);
36
-
37
- function useSidebar() {
38
- const context = React.useContext(SidebarContext);
39
- if (!context) {
40
- throw new Error('useSidebar must be used within a SidebarProvider.');
41
- }
42
-
43
- return context;
44
- }
45
-
46
- const SidebarProvider = React.forwardRef<
47
- HTMLDivElement,
48
- React.ComponentProps<'div'> & {
49
- defaultOpen?: boolean;
50
- open?: boolean;
51
- onOpenChange?: (open: boolean) => void;
52
- }
53
- >(
54
- (
55
- {
56
- defaultOpen = true,
57
- open: openProp,
58
- onOpenChange: setOpenProp,
59
- className,
60
- style,
61
- children,
62
- ...props
63
- },
64
- ref,
65
- ) => {
66
- const isMobile = useIsMobile();
67
- const [openMobile, setOpenMobile] = React.useState(false);
68
-
69
- // This is the internal state of the sidebar.
70
- // We use openProp and setOpenProp for control from outside the component.
71
- const [_open, _setOpen] = React.useState(defaultOpen);
72
- const open = openProp ?? _open;
73
- const setOpen = React.useCallback(
74
- (value: boolean | ((value: boolean) => boolean)) => {
75
- const openState = typeof value === 'function' ? value(open) : value;
76
- if (setOpenProp) {
77
- setOpenProp(openState);
78
- } else {
79
- _setOpen(openState);
80
- }
81
-
82
- // This sets the cookie to keep the sidebar state.
83
- document.cookie = `${SIDEBAR_COOKIE_NAME}=${openState}; path=/; max-age=${SIDEBAR_COOKIE_MAX_AGE}`;
84
- },
85
- [setOpenProp, open],
86
- );
87
-
88
- // Helper to toggle the sidebar.
89
- const toggleSidebar = React.useCallback(() => {
90
- return isMobile ? setOpenMobile(open => !open) : setOpen(open => !open);
91
- }, [isMobile, setOpen, setOpenMobile]);
92
-
93
- // Adds a keyboard shortcut to toggle the sidebar.
94
- React.useEffect(() => {
95
- const handleKeyDown = (event: KeyboardEvent) => {
96
- if (event.key === SIDEBAR_KEYBOARD_SHORTCUT && (event.metaKey || event.ctrlKey)) {
97
- event.preventDefault();
98
- toggleSidebar();
99
- }
100
- };
101
-
102
- window.addEventListener('keydown', handleKeyDown);
103
- return () => window.removeEventListener('keydown', handleKeyDown);
104
- }, [toggleSidebar]);
105
-
106
- // We add a state so that we can do data-state="expanded" or "collapsed".
107
- // This makes it easier to style the sidebar with Tailwind classes.
108
- const state = open ? 'expanded' : 'collapsed';
109
-
110
- const contextValue = React.useMemo<SidebarContext>(
111
- () => ({
112
- state,
113
- open,
114
- setOpen,
115
- isMobile,
116
- openMobile,
117
- setOpenMobile,
118
- toggleSidebar,
119
- }),
120
- [state, open, setOpen, isMobile, openMobile, setOpenMobile, toggleSidebar],
121
- );
122
-
123
- return (
124
- <SidebarContext.Provider value={contextValue}>
125
- <TooltipProvider delayDuration={0}>
126
- <div
127
- data-slot="sidebar-wrapper"
128
- style={
129
- {
130
- '--sidebar-width': SIDEBAR_WIDTH,
131
- '--sidebar-width-icon': SIDEBAR_WIDTH_ICON,
132
- ...style,
133
- } as React.CSSProperties
134
- }
135
- className={cn(
136
- 'group/sidebar-wrapper has-data-[variant=inset]:bg-sidebar flex min-h-svh w-full',
137
- className,
138
- )}
139
- ref={ref}
140
- {...props}
141
- >
142
- {children}
143
- </div>
144
- </TooltipProvider>
145
- </SidebarContext.Provider>
146
- );
147
- },
148
- );
149
- SidebarProvider.displayName = 'SidebarProvider';
150
-
151
- function Sidebar({
152
- side = 'left',
153
- variant = 'sidebar',
154
- collapsible = 'offcanvas',
155
- className,
156
- children,
157
- ...props
158
- }: React.ComponentProps<'div'> & {
159
- side?: 'left' | 'right';
160
- variant?: 'sidebar' | 'floating' | 'inset';
161
- collapsible?: 'offcanvas' | 'icon' | 'none';
162
- }) {
163
- const { isMobile, state, openMobile, setOpenMobile } = useSidebar();
164
-
165
- if (collapsible === 'none') {
166
- return (
167
- <div
168
- data-slot="sidebar"
169
- className={cn(
170
- 'bg-sidebar text-sidebar-foreground flex h-full w-(--sidebar-width) flex-col',
171
- className,
172
- )}
173
- {...props}
174
- >
175
- {children}
176
- </div>
177
- );
178
- }
179
-
180
- if (isMobile) {
181
- return (
182
- <Sheet open={openMobile} onOpenChange={setOpenMobile} {...props}>
183
- <SheetHeader className="sr-only">
184
- <SheetTitle>Sidebar</SheetTitle>
185
- <SheetDescription>Displays the mobile sidebar.</SheetDescription>
186
- </SheetHeader>
187
- <SheetContent
188
- data-sidebar="sidebar"
189
- data-slot="sidebar"
190
- data-mobile="true"
191
- className="bg-sidebar text-sidebar-foreground w-(--sidebar-width) p-0 [&>button]:hidden"
192
- style={
193
- {
194
- '--sidebar-width': SIDEBAR_WIDTH_MOBILE,
195
- } as React.CSSProperties
196
- }
197
- side={side}
198
- >
199
- <div className="flex h-full w-full flex-col">{children}</div>
200
- </SheetContent>
201
- </Sheet>
202
- );
203
- }
204
-
205
- return (
206
- <div
207
- className="group peer text-sidebar-foreground hidden md:block"
208
- data-state={state}
209
- data-collapsible={state === 'collapsed' ? collapsible : ''}
210
- data-variant={variant}
211
- data-side={side}
212
- data-slot="sidebar"
213
- >
214
- {/* This is what handles the sidebar gap on desktop */}
215
- <div
216
- className={cn(
217
- 'relative h-svh w-(--sidebar-width) bg-transparent transition-[width] duration-200 ease-linear',
218
- 'group-data-[collapsible=offcanvas]:w-0',
219
- 'group-data-[side=right]:rotate-180',
220
- variant === 'floating' || variant === 'inset'
221
- ? 'group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4)))]'
222
- : 'group-data-[collapsible=icon]:w-(--sidebar-width-icon)',
223
- )}
224
- />
225
- <div
226
- className={cn(
227
- 'fixed inset-y-0 z-10 hidden h-svh w-(--sidebar-width) transition-[left,right,width] duration-200 ease-linear md:flex',
228
- side === 'left'
229
- ? 'left-0 group-data-[collapsible=offcanvas]:left-[calc(var(--sidebar-width)*-1)]'
230
- : 'right-0 group-data-[collapsible=offcanvas]:right-[calc(var(--sidebar-width)*-1)]',
231
- // Adjust the padding for floating and inset variants.
232
- variant === 'floating' || variant === 'inset'
233
- ? 'p-2 group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4))+2px)]'
234
- : 'group-data-[collapsible=icon]:w-(--sidebar-width-icon) group-data-[side=left]:border-r group-data-[side=right]:border-l',
235
- className,
236
- )}
237
- {...props}
238
- >
239
- <div
240
- data-sidebar="sidebar"
241
- className="bg-sidebar group-data-[variant=floating]:border-sidebar-border flex h-full w-full flex-col group-data-[variant=floating]:rounded-lg group-data-[variant=floating]:border group-data-[variant=floating]:shadow-sm"
242
- >
243
- {children}
244
- </div>
245
- </div>
246
- </div>
247
- );
248
- }
249
-
250
- function SidebarTrigger({ className, onClick, ...props }: React.ComponentProps<typeof Button>) {
251
- const { toggleSidebar } = useSidebar();
252
-
253
- return (
254
- <Button
255
- data-sidebar="trigger"
256
- data-slot="sidebar-trigger"
257
- variant="ghost"
258
- size="icon"
259
- className={cn('h-7 w-7', className)}
260
- onClick={(event: MouseEvent<any>) => {
261
- onClick?.(event);
262
- toggleSidebar();
263
- }}
264
- {...props}
265
- >
266
- <PanelLeftIcon />
267
- <span className="sr-only">Toggle Sidebar</span>
268
- </Button>
269
- );
270
- }
271
-
272
- function SidebarRail({ className, ...props }: React.ComponentProps<'button'>) {
273
- const { toggleSidebar } = useSidebar();
274
-
275
- return (
276
- <button
277
- data-sidebar="rail"
278
- data-slot="sidebar-rail"
279
- aria-label="Toggle Sidebar"
280
- tabIndex={-1}
281
- onClick={toggleSidebar}
282
- title="Toggle Sidebar"
283
- className={cn(
284
- 'hover:after:bg-sidebar-border absolute inset-y-0 z-20 hidden w-4 -translate-x-1/2 transition-all ease-linear group-data-[side=left]:-right-4 group-data-[side=right]:left-0 after:absolute after:inset-y-0 after:left-1/2 after:w-[2px] sm:flex',
285
- 'in-data-[side=left]:cursor-w-resize in-data-[side=right]:cursor-e-resize',
286
- '[[data-side=left][data-state=collapsed]_&]:cursor-e-resize [[data-side=right][data-state=collapsed]_&]:cursor-w-resize',
287
- 'hover:group-data-[collapsible=offcanvas]:bg-sidebar group-data-[collapsible=offcanvas]:translate-x-0 group-data-[collapsible=offcanvas]:after:left-full',
288
- '[[data-side=left][data-collapsible=offcanvas]_&]:-right-2',
289
- '[[data-side=right][data-collapsible=offcanvas]_&]:-left-2',
290
- className,
291
- )}
292
- {...props}
293
- />
294
- );
295
- }
296
-
297
- function SidebarInset({ className, ...props }: React.ComponentProps<'main'>) {
298
- return (
299
- <main
300
- data-slot="sidebar-inset"
301
- className={cn(
302
- 'bg-background relative flex min-h-svh flex-1 flex-col',
303
- 'peer-data-[variant=inset]:min-h-[calc(100svh-(--spacing(4)))] md:peer-data-[variant=inset]:m-2 md:peer-data-[variant=inset]:ml-0 md:peer-data-[variant=inset]:rounded-xl md:peer-data-[variant=inset]:shadow-sm md:peer-data-[variant=inset]:peer-data-[state=collapsed]:ml-2',
304
- className,
305
- )}
306
- {...props}
307
- />
308
- );
309
- }
310
-
311
- function SidebarInput({ className, ...props }: React.ComponentProps<typeof Input>) {
312
- return (
313
- <Input
314
- data-slot="sidebar-input"
315
- data-sidebar="input"
316
- className={cn('bg-background h-8 w-full shadow-none', className)}
317
- {...props}
318
- />
319
- );
320
- }
321
-
322
- function SidebarHeader({ className, ...props }: React.ComponentProps<'div'>) {
323
- return (
324
- <div
325
- data-slot="sidebar-header"
326
- data-sidebar="header"
327
- className={cn('flex flex-col gap-2 p-2', className)}
328
- {...props}
329
- />
330
- );
331
- }
332
-
333
- function SidebarFooter({ className, ...props }: React.ComponentProps<'div'>) {
334
- return (
335
- <div
336
- data-slot="sidebar-footer"
337
- data-sidebar="footer"
338
- className={cn('flex flex-col gap-2 p-2', className)}
339
- {...props}
340
- />
341
- );
342
- }
343
-
344
- function SidebarSeparator({ className, ...props }: React.ComponentProps<typeof Separator>) {
345
- return (
346
- <Separator
347
- data-slot="sidebar-separator"
348
- data-sidebar="separator"
349
- className={cn('bg-sidebar-border mx-2 w-auto', className)}
350
- {...props}
351
- />
352
- );
353
- }
354
-
355
- function SidebarContent({ className, ...props }: React.ComponentProps<'div'>) {
356
- return (
357
- <div
358
- data-slot="sidebar-content"
359
- data-sidebar="content"
360
- className={cn(
361
- 'flex min-h-0 flex-1 flex-col gap-2 overflow-auto group-data-[collapsible=icon]:overflow-hidden',
362
- className,
363
- )}
364
- {...props}
365
- />
366
- );
367
- }
368
-
369
- function SidebarGroup({ className, ...props }: React.ComponentProps<'div'>) {
370
- return (
371
- <div
372
- data-slot="sidebar-group"
373
- data-sidebar="group"
374
- className={cn('relative flex w-full min-w-0 flex-col p-2', className)}
375
- {...props}
376
- />
377
- );
378
- }
379
-
380
- function SidebarGroupLabel({
381
- className,
382
- asChild = false,
383
- ...props
384
- }: React.ComponentProps<'div'> & { asChild?: boolean }) {
385
- const Comp = asChild ? Slot : 'div';
386
-
387
- return (
388
- <Comp
389
- data-slot="sidebar-group-label"
390
- data-sidebar="group-label"
391
- className={cn(
392
- 'text-sidebar-foreground/70 ring-sidebar-ring flex h-8 shrink-0 items-center rounded-md px-2 text-xs font-medium outline-hidden transition-[margin,opa] duration-200 ease-linear focus-visible:ring-2 [&>svg]:size-4 [&>svg]:shrink-0',
393
- 'group-data-[collapsible=icon]:-mt-8 group-data-[collapsible=icon]:opacity-0',
394
- className,
395
- )}
396
- {...props}
397
- />
398
- );
399
- }
400
-
401
- function SidebarGroupAction({
402
- className,
403
- asChild = false,
404
- ...props
405
- }: React.ComponentProps<'button'> & { asChild?: boolean }) {
406
- const Comp = asChild ? Slot : 'button';
407
-
408
- return (
409
- <Comp
410
- data-slot="sidebar-group-action"
411
- data-sidebar="group-action"
412
- className={cn(
413
- 'text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground absolute top-3.5 right-3 flex aspect-square w-5 items-center justify-center rounded-md p-0 outline-hidden transition-transform focus-visible:ring-2 [&>svg]:size-4 [&>svg]:shrink-0',
414
- // Increases the hit area of the button on mobile.
415
- 'after:absolute after:-inset-2 md:after:hidden',
416
- 'group-data-[collapsible=icon]:hidden',
417
- className,
418
- )}
419
- {...props}
420
- />
421
- );
422
- }
423
-
424
- function SidebarGroupContent({ className, ...props }: React.ComponentProps<'div'>) {
425
- return (
426
- <div
427
- data-slot="sidebar-group-content"
428
- data-sidebar="group-content"
429
- className={cn('w-full text-sm', className)}
430
- {...props}
431
- />
432
- );
433
- }
434
-
435
- function SidebarMenu({ className, ...props }: React.ComponentProps<'ul'>) {
436
- return (
437
- <ul
438
- data-slot="sidebar-menu"
439
- data-sidebar="menu"
440
- className={cn('flex w-full min-w-0 flex-col gap-1', className)}
441
- {...props}
442
- />
443
- );
444
- }
445
-
446
- function SidebarMenuItem({ className, ...props }: React.ComponentProps<'li'>) {
447
- return (
448
- <li
449
- data-slot="sidebar-menu-item"
450
- data-sidebar="menu-item"
451
- className={cn('group/menu-item relative', className)}
452
- {...props}
453
- />
454
- );
455
- }
456
-
457
- const sidebarMenuButtonVariants = cva(
458
- 'peer/menu-button flex w-full items-center gap-2 overflow-hidden rounded-md p-2 text-left text-sm outline-hidden ring-sidebar-ring transition-[width,height,padding] hover:bg-sidebar-accent hover:text-sidebar-accent-foreground focus-visible:ring-2 active:bg-sidebar-accent active:text-sidebar-accent-foreground disabled:pointer-events-none disabled:opacity-50 group-has-data-[sidebar=menu-action]/menu-item:pr-8 aria-disabled:pointer-events-none aria-disabled:opacity-50 data-[active=true]:bg-sidebar-accent data-[active=true]:font-medium data-[active=true]:text-sidebar-accent-foreground data-[state=open]:hover:bg-sidebar-accent data-[state=open]:hover:text-sidebar-accent-foreground group-data-[collapsible=icon]:size-8! group-data-[collapsible=icon]:p-2! [&>span:last-child]:truncate [&>svg]:size-4 [&>svg]:shrink-0',
459
- {
460
- variants: {
461
- variant: {
462
- default: 'hover:bg-sidebar-accent hover:text-sidebar-accent-foreground',
463
- outline:
464
- 'bg-background shadow-[0_0_0_1px_hsl(var(--sidebar-border))] hover:bg-sidebar-accent hover:text-sidebar-accent-foreground hover:shadow-[0_0_0_1px_hsl(var(--sidebar-accent))]',
465
- },
466
- size: {
467
- default: 'h-8 text-sm',
468
- sm: 'h-7 text-xs',
469
- lg: 'h-12 text-sm group-data-[collapsible=icon]:p-0!',
470
- },
471
- },
472
- defaultVariants: {
473
- variant: 'default',
474
- size: 'default',
475
- },
476
- },
477
- );
478
-
479
- function SidebarMenuButton({
480
- asChild = false,
481
- isActive = false,
482
- variant = 'default',
483
- size = 'default',
484
- tooltip,
485
- className,
486
- ...props
487
- }: React.ComponentProps<'button'> & {
488
- asChild?: boolean;
489
- isActive?: boolean;
490
- tooltip?: string | React.ComponentProps<typeof TooltipContent>;
491
- } & VariantProps<typeof sidebarMenuButtonVariants>) {
492
- const Comp = asChild ? Slot : 'button';
493
- const { isMobile, state } = useSidebar();
494
-
495
- const button = (
496
- <Comp
497
- data-slot="sidebar-menu-button"
498
- data-sidebar="menu-button"
499
- data-size={size}
500
- data-active={isActive}
501
- className={cn(sidebarMenuButtonVariants({ variant, size }), className)}
502
- {...props}
503
- />
504
- );
505
-
506
- if (!tooltip) {
507
- return button;
508
- }
509
-
510
- if (typeof tooltip === 'string') {
511
- tooltip = {
512
- children: tooltip,
513
- };
514
- }
515
-
516
- return (
517
- <Tooltip>
518
- <TooltipTrigger asChild>{button}</TooltipTrigger>
519
- <TooltipContent
520
- side="right"
521
- align="center"
522
- hidden={state !== 'collapsed' || isMobile}
523
- {...tooltip}
524
- />
525
- </Tooltip>
526
- );
527
- }
528
-
529
- function SidebarMenuAction({
530
- className,
531
- asChild = false,
532
- showOnHover = false,
533
- ...props
534
- }: React.ComponentProps<'button'> & {
535
- asChild?: boolean;
536
- showOnHover?: boolean;
537
- }) {
538
- const Comp = asChild ? Slot : 'button';
539
-
540
- return (
541
- <Comp
542
- data-slot="sidebar-menu-action"
543
- data-sidebar="menu-action"
544
- className={cn(
545
- 'text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground peer-hover/menu-button:text-sidebar-accent-foreground absolute top-1.5 right-1 flex aspect-square w-5 items-center justify-center rounded-md p-0 outline-hidden transition-transform focus-visible:ring-2 [&>svg]:size-4 [&>svg]:shrink-0',
546
- // Increases the hit area of the button on mobile.
547
- 'after:absolute after:-inset-2 md:after:hidden',
548
- 'peer-data-[size=sm]/menu-button:top-1',
549
- 'peer-data-[size=default]/menu-button:top-1.5',
550
- 'peer-data-[size=lg]/menu-button:top-2.5',
551
- 'group-data-[collapsible=icon]:hidden',
552
- showOnHover &&
553
- 'peer-data-[active=true]/menu-button:text-sidebar-accent-foreground group-focus-within/menu-item:opacity-100 group-hover/menu-item:opacity-100 data-[state=open]:opacity-100 md:opacity-0',
554
- className,
555
- )}
556
- {...props}
557
- />
558
- );
559
- }
560
-
561
- function SidebarMenuBadge({ className, ...props }: React.ComponentProps<'div'>) {
562
- return (
563
- <div
564
- data-slot="sidebar-menu-badge"
565
- data-sidebar="menu-badge"
566
- className={cn(
567
- 'text-sidebar-foreground pointer-events-none absolute right-1 flex h-5 min-w-5 items-center justify-center rounded-md px-1 text-xs font-medium tabular-nums select-none',
568
- 'peer-hover/menu-button:text-sidebar-accent-foreground peer-data-[active=true]/menu-button:text-sidebar-accent-foreground',
569
- 'peer-data-[size=sm]/menu-button:top-1',
570
- 'peer-data-[size=default]/menu-button:top-1.5',
571
- 'peer-data-[size=lg]/menu-button:top-2.5',
572
- 'group-data-[collapsible=icon]:hidden',
573
- className,
574
- )}
575
- {...props}
576
- />
577
- );
578
- }
579
-
580
- function SidebarMenuSkeleton({
581
- className,
582
- showIcon = false,
583
- ...props
584
- }: React.ComponentProps<'div'> & {
585
- showIcon?: boolean;
586
- }) {
587
- // Random width between 50 to 90%.
588
- const width = React.useMemo(() => {
589
- return `${Math.floor(Math.random() * 40) + 50}%`;
590
- }, []);
591
-
592
- return (
593
- <div
594
- data-slot="sidebar-menu-skeleton"
595
- data-sidebar="menu-skeleton"
596
- className={cn('flex h-8 items-center gap-2 rounded-md px-2', className)}
597
- {...props}
598
- >
599
- {showIcon && <Skeleton className="size-4 rounded-md" data-sidebar="menu-skeleton-icon" />}
600
- <Skeleton
601
- className="h-4 max-w-(--skeleton-width) flex-1"
602
- data-sidebar="menu-skeleton-text"
603
- style={
604
- {
605
- '--skeleton-width': width,
606
- } as React.CSSProperties
607
- }
608
- />
609
- </div>
610
- );
611
- }
612
-
613
- function SidebarMenuSub({ className, ...props }: React.ComponentProps<'ul'>) {
614
- return (
615
- <ul
616
- data-slot="sidebar-menu-sub"
617
- data-sidebar="menu-sub"
618
- className={cn(
619
- 'border-sidebar-border mx-3.5 flex min-w-0 translate-x-px flex-col gap-1 border-l px-2.5 py-0.5',
620
- 'group-data-[collapsible=icon]:hidden',
621
- className,
622
- )}
623
- {...props}
624
- />
625
- );
626
- }
627
-
628
- function SidebarMenuSubItem({ className, ...props }: React.ComponentProps<'li'>) {
629
- return (
630
- <li
631
- data-slot="sidebar-menu-sub-item"
632
- data-sidebar="menu-sub-item"
633
- className={cn('group/menu-sub-item relative', className)}
634
- {...props}
635
- />
636
- );
637
- }
638
-
639
- function SidebarMenuSubButton({
640
- asChild = false,
641
- size = 'md',
642
- isActive = false,
643
- className,
644
- ...props
645
- }: React.ComponentProps<'a'> & {
646
- asChild?: boolean;
647
- size?: 'sm' | 'md';
648
- isActive?: boolean;
649
- }) {
650
- const Comp = asChild ? Slot : 'a';
651
-
652
- return (
653
- <Comp
654
- data-slot="sidebar-menu-sub-button"
655
- data-sidebar="menu-sub-button"
656
- data-size={size}
657
- data-active={isActive}
658
- className={cn(
659
- 'text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground active:bg-sidebar-accent active:text-sidebar-accent-foreground [&>svg]:text-sidebar-accent-foreground flex h-7 min-w-0 -translate-x-px items-center gap-2 overflow-hidden rounded-md px-2 outline-hidden focus-visible:ring-2 disabled:pointer-events-none disabled:opacity-50 aria-disabled:pointer-events-none aria-disabled:opacity-50 [&>span:last-child]:truncate [&>svg]:size-4 [&>svg]:shrink-0',
660
- 'data-[active=true]:bg-sidebar-accent data-[active=true]:text-sidebar-accent-foreground',
661
- size === 'sm' && 'text-xs',
662
- size === 'md' && 'text-sm',
663
- 'group-data-[collapsible=icon]:hidden',
664
- className,
665
- )}
666
- {...props}
667
- />
668
- );
669
- }
670
-
671
- export {
672
- Sidebar,
673
- SidebarContent,
674
- SidebarFooter,
675
- SidebarGroup,
676
- SidebarGroupAction,
677
- SidebarGroupContent,
678
- SidebarGroupLabel,
679
- SidebarHeader,
680
- SidebarInput,
681
- SidebarInset,
682
- SidebarMenu,
683
- SidebarMenuAction,
684
- SidebarMenuBadge,
685
- SidebarMenuButton,
686
- SidebarMenuItem,
687
- SidebarMenuSkeleton,
688
- SidebarMenuSub,
689
- SidebarMenuSubButton,
690
- SidebarMenuSubItem,
691
- SidebarProvider,
692
- SidebarRail,
693
- SidebarSeparator,
694
- SidebarTrigger,
695
- useSidebar,
696
- };
1
+ 'use client';
2
+
3
+ import { MouseEvent } from 'react';
4
+ import * as React from 'react';
5
+ import { Slot } from '@radix-ui/react-slot';
6
+ import { VariantProps, cva } from 'class-variance-authority';
7
+ import { PanelLeftIcon } from 'lucide-react';
8
+
9
+ import { useIsMobile } from '@/hooks/use-mobile.js';
10
+ import { cn } from '@/lib/utils.js';
11
+ import { Button } from '@/components/ui/button.js';
12
+ import { Input } from '@/components/ui/input.js';
13
+ import { Separator } from '@/components/ui/separator.js';
14
+ import { Sheet, SheetContent, SheetDescription, SheetHeader, SheetTitle } from '@/components/ui/sheet.js';
15
+ import { Skeleton } from '@/components/ui/skeleton.js';
16
+ import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip.js';
17
+
18
+ const SIDEBAR_COOKIE_NAME = 'sidebar_state';
19
+ const SIDEBAR_COOKIE_MAX_AGE = 60 * 60 * 24 * 7;
20
+ const SIDEBAR_WIDTH = '16rem';
21
+ const SIDEBAR_WIDTH_MOBILE = '18rem';
22
+ const SIDEBAR_WIDTH_ICON = '3rem';
23
+ const SIDEBAR_KEYBOARD_SHORTCUT = 'b';
24
+
25
+ type SidebarContext = {
26
+ state: 'expanded' | 'collapsed';
27
+ open: boolean;
28
+ setOpen: (open: boolean) => void;
29
+ openMobile: boolean;
30
+ setOpenMobile: (open: boolean) => void;
31
+ isMobile: boolean;
32
+ toggleSidebar: () => void;
33
+ };
34
+
35
+ const SidebarContext = React.createContext<SidebarContext | null>(null);
36
+
37
+ function useSidebar() {
38
+ const context = React.useContext(SidebarContext);
39
+ if (!context) {
40
+ throw new Error('useSidebar must be used within a SidebarProvider.');
41
+ }
42
+
43
+ return context;
44
+ }
45
+
46
+ const SidebarProvider = React.forwardRef<
47
+ HTMLDivElement,
48
+ React.ComponentProps<'div'> & {
49
+ defaultOpen?: boolean;
50
+ open?: boolean;
51
+ onOpenChange?: (open: boolean) => void;
52
+ }
53
+ >(
54
+ (
55
+ {
56
+ defaultOpen = true,
57
+ open: openProp,
58
+ onOpenChange: setOpenProp,
59
+ className,
60
+ style,
61
+ children,
62
+ ...props
63
+ },
64
+ ref,
65
+ ) => {
66
+ const isMobile = useIsMobile();
67
+ const [openMobile, setOpenMobile] = React.useState(false);
68
+
69
+ // This is the internal state of the sidebar.
70
+ // We use openProp and setOpenProp for control from outside the component.
71
+ const [_open, _setOpen] = React.useState(defaultOpen);
72
+ const open = openProp ?? _open;
73
+ const setOpen = React.useCallback(
74
+ (value: boolean | ((value: boolean) => boolean)) => {
75
+ const openState = typeof value === 'function' ? value(open) : value;
76
+ if (setOpenProp) {
77
+ setOpenProp(openState);
78
+ } else {
79
+ _setOpen(openState);
80
+ }
81
+
82
+ // This sets the cookie to keep the sidebar state.
83
+ document.cookie = `${SIDEBAR_COOKIE_NAME}=${openState}; path=/; max-age=${SIDEBAR_COOKIE_MAX_AGE}`;
84
+ },
85
+ [setOpenProp, open],
86
+ );
87
+
88
+ // Helper to toggle the sidebar.
89
+ const toggleSidebar = React.useCallback(() => {
90
+ return isMobile ? setOpenMobile(open => !open) : setOpen(open => !open);
91
+ }, [isMobile, setOpen, setOpenMobile]);
92
+
93
+ // Adds a keyboard shortcut to toggle the sidebar.
94
+ React.useEffect(() => {
95
+ const handleKeyDown = (event: KeyboardEvent) => {
96
+ if (event.key === SIDEBAR_KEYBOARD_SHORTCUT && (event.metaKey || event.ctrlKey)) {
97
+ event.preventDefault();
98
+ toggleSidebar();
99
+ }
100
+ };
101
+
102
+ window.addEventListener('keydown', handleKeyDown);
103
+ return () => window.removeEventListener('keydown', handleKeyDown);
104
+ }, [toggleSidebar]);
105
+
106
+ // We add a state so that we can do data-state="expanded" or "collapsed".
107
+ // This makes it easier to style the sidebar with Tailwind classes.
108
+ const state = open ? 'expanded' : 'collapsed';
109
+
110
+ const contextValue = React.useMemo<SidebarContext>(
111
+ () => ({
112
+ state,
113
+ open,
114
+ setOpen,
115
+ isMobile,
116
+ openMobile,
117
+ setOpenMobile,
118
+ toggleSidebar,
119
+ }),
120
+ [state, open, setOpen, isMobile, openMobile, setOpenMobile, toggleSidebar],
121
+ );
122
+
123
+ return (
124
+ <SidebarContext.Provider value={contextValue}>
125
+ <TooltipProvider delayDuration={0}>
126
+ <div
127
+ data-slot="sidebar-wrapper"
128
+ style={
129
+ {
130
+ '--sidebar-width': SIDEBAR_WIDTH,
131
+ '--sidebar-width-icon': SIDEBAR_WIDTH_ICON,
132
+ ...style,
133
+ } as React.CSSProperties
134
+ }
135
+ className={cn(
136
+ 'group/sidebar-wrapper has-data-[variant=inset]:bg-sidebar flex min-h-svh w-full',
137
+ className,
138
+ )}
139
+ ref={ref}
140
+ {...props}
141
+ >
142
+ {children}
143
+ </div>
144
+ </TooltipProvider>
145
+ </SidebarContext.Provider>
146
+ );
147
+ },
148
+ );
149
+ SidebarProvider.displayName = 'SidebarProvider';
150
+
151
+ function Sidebar({
152
+ side = 'left',
153
+ variant = 'sidebar',
154
+ collapsible = 'offcanvas',
155
+ className,
156
+ children,
157
+ ...props
158
+ }: React.ComponentProps<'div'> & {
159
+ side?: 'left' | 'right';
160
+ variant?: 'sidebar' | 'floating' | 'inset';
161
+ collapsible?: 'offcanvas' | 'icon' | 'none';
162
+ }) {
163
+ const { isMobile, state, openMobile, setOpenMobile } = useSidebar();
164
+
165
+ if (collapsible === 'none') {
166
+ return (
167
+ <div
168
+ data-slot="sidebar"
169
+ className={cn(
170
+ 'bg-sidebar text-sidebar-foreground flex h-full w-(--sidebar-width) flex-col',
171
+ className,
172
+ )}
173
+ {...props}
174
+ >
175
+ {children}
176
+ </div>
177
+ );
178
+ }
179
+
180
+ if (isMobile) {
181
+ return (
182
+ <Sheet open={openMobile} onOpenChange={setOpenMobile} {...props}>
183
+ <SheetHeader className="sr-only">
184
+ <SheetTitle>Sidebar</SheetTitle>
185
+ <SheetDescription>Displays the mobile sidebar.</SheetDescription>
186
+ </SheetHeader>
187
+ <SheetContent
188
+ data-sidebar="sidebar"
189
+ data-slot="sidebar"
190
+ data-mobile="true"
191
+ className="bg-sidebar text-sidebar-foreground w-(--sidebar-width) p-0 [&>button]:hidden"
192
+ style={
193
+ {
194
+ '--sidebar-width': SIDEBAR_WIDTH_MOBILE,
195
+ } as React.CSSProperties
196
+ }
197
+ side={side}
198
+ >
199
+ <div className="flex h-full w-full flex-col">{children}</div>
200
+ </SheetContent>
201
+ </Sheet>
202
+ );
203
+ }
204
+
205
+ return (
206
+ <div
207
+ className="group peer text-sidebar-foreground hidden md:block"
208
+ data-state={state}
209
+ data-collapsible={state === 'collapsed' ? collapsible : ''}
210
+ data-variant={variant}
211
+ data-side={side}
212
+ data-slot="sidebar"
213
+ >
214
+ {/* This is what handles the sidebar gap on desktop */}
215
+ <div
216
+ className={cn(
217
+ 'relative h-svh w-(--sidebar-width) bg-transparent transition-[width] duration-200 ease-linear',
218
+ 'group-data-[collapsible=offcanvas]:w-0',
219
+ 'group-data-[side=right]:rotate-180',
220
+ variant === 'floating' || variant === 'inset'
221
+ ? 'group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4)))]'
222
+ : 'group-data-[collapsible=icon]:w-(--sidebar-width-icon)',
223
+ )}
224
+ />
225
+ <div
226
+ className={cn(
227
+ 'fixed inset-y-0 z-10 hidden h-svh w-(--sidebar-width) transition-[left,right,width] duration-200 ease-linear md:flex',
228
+ side === 'left'
229
+ ? 'left-0 group-data-[collapsible=offcanvas]:left-[calc(var(--sidebar-width)*-1)]'
230
+ : 'right-0 group-data-[collapsible=offcanvas]:right-[calc(var(--sidebar-width)*-1)]',
231
+ // Adjust the padding for floating and inset variants.
232
+ variant === 'floating' || variant === 'inset'
233
+ ? 'p-2 group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4))+2px)]'
234
+ : 'group-data-[collapsible=icon]:w-(--sidebar-width-icon) group-data-[side=left]:border-r group-data-[side=right]:border-l',
235
+ className,
236
+ )}
237
+ {...props}
238
+ >
239
+ <div
240
+ data-sidebar="sidebar"
241
+ className="bg-sidebar group-data-[variant=floating]:border-sidebar-border flex h-full w-full flex-col group-data-[variant=floating]:rounded-lg group-data-[variant=floating]:border group-data-[variant=floating]:shadow-sm"
242
+ >
243
+ {children}
244
+ </div>
245
+ </div>
246
+ </div>
247
+ );
248
+ }
249
+
250
+ function SidebarTrigger({ className, onClick, ...props }: React.ComponentProps<typeof Button>) {
251
+ const { toggleSidebar } = useSidebar();
252
+
253
+ return (
254
+ <Button
255
+ data-sidebar="trigger"
256
+ data-slot="sidebar-trigger"
257
+ variant="ghost"
258
+ size="icon"
259
+ className={cn('h-7 w-7', className)}
260
+ onClick={(event: MouseEvent<any>) => {
261
+ onClick?.(event);
262
+ toggleSidebar();
263
+ }}
264
+ {...props}
265
+ >
266
+ <PanelLeftIcon />
267
+ <span className="sr-only">Toggle Sidebar</span>
268
+ </Button>
269
+ );
270
+ }
271
+
272
+ function SidebarRail({ className, ...props }: React.ComponentProps<'button'>) {
273
+ const { toggleSidebar } = useSidebar();
274
+
275
+ return (
276
+ <button
277
+ data-sidebar="rail"
278
+ data-slot="sidebar-rail"
279
+ aria-label="Toggle Sidebar"
280
+ tabIndex={-1}
281
+ onClick={toggleSidebar}
282
+ title="Toggle Sidebar"
283
+ className={cn(
284
+ 'hover:after:bg-sidebar-border absolute inset-y-0 z-20 hidden w-4 -translate-x-1/2 transition-all ease-linear group-data-[side=left]:-right-4 group-data-[side=right]:left-0 after:absolute after:inset-y-0 after:left-1/2 after:w-[2px] sm:flex',
285
+ 'in-data-[side=left]:cursor-w-resize in-data-[side=right]:cursor-e-resize',
286
+ '[[data-side=left][data-state=collapsed]_&]:cursor-e-resize [[data-side=right][data-state=collapsed]_&]:cursor-w-resize',
287
+ 'hover:group-data-[collapsible=offcanvas]:bg-sidebar group-data-[collapsible=offcanvas]:translate-x-0 group-data-[collapsible=offcanvas]:after:left-full',
288
+ '[[data-side=left][data-collapsible=offcanvas]_&]:-right-2',
289
+ '[[data-side=right][data-collapsible=offcanvas]_&]:-left-2',
290
+ className,
291
+ )}
292
+ {...props}
293
+ />
294
+ );
295
+ }
296
+
297
+ function SidebarInset({ className, ...props }: React.ComponentProps<'main'>) {
298
+ return (
299
+ <main
300
+ data-slot="sidebar-inset"
301
+ className={cn(
302
+ 'bg-background relative flex min-h-svh flex-1 flex-col',
303
+ 'peer-data-[variant=inset]:min-h-[calc(100svh-(--spacing(4)))] md:peer-data-[variant=inset]:m-2 md:peer-data-[variant=inset]:ml-0 md:peer-data-[variant=inset]:rounded-xl md:peer-data-[variant=inset]:shadow-sm md:peer-data-[variant=inset]:peer-data-[state=collapsed]:ml-2',
304
+ className,
305
+ )}
306
+ {...props}
307
+ />
308
+ );
309
+ }
310
+
311
+ function SidebarInput({ className, ...props }: React.ComponentProps<typeof Input>) {
312
+ return (
313
+ <Input
314
+ data-slot="sidebar-input"
315
+ data-sidebar="input"
316
+ className={cn('bg-background h-8 w-full shadow-none', className)}
317
+ {...props}
318
+ />
319
+ );
320
+ }
321
+
322
+ function SidebarHeader({ className, ...props }: React.ComponentProps<'div'>) {
323
+ return (
324
+ <div
325
+ data-slot="sidebar-header"
326
+ data-sidebar="header"
327
+ className={cn('flex flex-col gap-2 p-2', className)}
328
+ {...props}
329
+ />
330
+ );
331
+ }
332
+
333
+ function SidebarFooter({ className, ...props }: React.ComponentProps<'div'>) {
334
+ return (
335
+ <div
336
+ data-slot="sidebar-footer"
337
+ data-sidebar="footer"
338
+ className={cn('flex flex-col gap-2 p-2', className)}
339
+ {...props}
340
+ />
341
+ );
342
+ }
343
+
344
+ function SidebarSeparator({ className, ...props }: React.ComponentProps<typeof Separator>) {
345
+ return (
346
+ <Separator
347
+ data-slot="sidebar-separator"
348
+ data-sidebar="separator"
349
+ className={cn('bg-sidebar-border mx-2 w-auto', className)}
350
+ {...props}
351
+ />
352
+ );
353
+ }
354
+
355
+ function SidebarContent({ className, ...props }: React.ComponentProps<'div'>) {
356
+ return (
357
+ <div
358
+ data-slot="sidebar-content"
359
+ data-sidebar="content"
360
+ className={cn(
361
+ 'flex min-h-0 flex-1 flex-col gap-2 overflow-auto group-data-[collapsible=icon]:overflow-hidden',
362
+ className,
363
+ )}
364
+ {...props}
365
+ />
366
+ );
367
+ }
368
+
369
+ function SidebarGroup({ className, ...props }: React.ComponentProps<'div'>) {
370
+ return (
371
+ <div
372
+ data-slot="sidebar-group"
373
+ data-sidebar="group"
374
+ className={cn('relative flex w-full min-w-0 flex-col p-2', className)}
375
+ {...props}
376
+ />
377
+ );
378
+ }
379
+
380
+ function SidebarGroupLabel({
381
+ className,
382
+ asChild = false,
383
+ ...props
384
+ }: React.ComponentProps<'div'> & { asChild?: boolean }) {
385
+ const Comp = asChild ? Slot : 'div';
386
+
387
+ return (
388
+ <Comp
389
+ data-slot="sidebar-group-label"
390
+ data-sidebar="group-label"
391
+ className={cn(
392
+ 'text-sidebar-foreground/70 ring-sidebar-ring flex h-8 shrink-0 items-center rounded-md px-2 text-xs font-medium outline-hidden transition-[margin,opa] duration-200 ease-linear focus-visible:ring-2 [&>svg]:size-4 [&>svg]:shrink-0',
393
+ 'group-data-[collapsible=icon]:-mt-8 group-data-[collapsible=icon]:opacity-0',
394
+ className,
395
+ )}
396
+ {...props}
397
+ />
398
+ );
399
+ }
400
+
401
+ function SidebarGroupAction({
402
+ className,
403
+ asChild = false,
404
+ ...props
405
+ }: React.ComponentProps<'button'> & { asChild?: boolean }) {
406
+ const Comp = asChild ? Slot : 'button';
407
+
408
+ return (
409
+ <Comp
410
+ data-slot="sidebar-group-action"
411
+ data-sidebar="group-action"
412
+ className={cn(
413
+ 'text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground absolute top-3.5 right-3 flex aspect-square w-5 items-center justify-center rounded-md p-0 outline-hidden transition-transform focus-visible:ring-2 [&>svg]:size-4 [&>svg]:shrink-0',
414
+ // Increases the hit area of the button on mobile.
415
+ 'after:absolute after:-inset-2 md:after:hidden',
416
+ 'group-data-[collapsible=icon]:hidden',
417
+ className,
418
+ )}
419
+ {...props}
420
+ />
421
+ );
422
+ }
423
+
424
+ function SidebarGroupContent({ className, ...props }: React.ComponentProps<'div'>) {
425
+ return (
426
+ <div
427
+ data-slot="sidebar-group-content"
428
+ data-sidebar="group-content"
429
+ className={cn('w-full text-sm', className)}
430
+ {...props}
431
+ />
432
+ );
433
+ }
434
+
435
+ function SidebarMenu({ className, ...props }: React.ComponentProps<'ul'>) {
436
+ return (
437
+ <ul
438
+ data-slot="sidebar-menu"
439
+ data-sidebar="menu"
440
+ className={cn('flex w-full min-w-0 flex-col gap-1', className)}
441
+ {...props}
442
+ />
443
+ );
444
+ }
445
+
446
+ function SidebarMenuItem({ className, ...props }: React.ComponentProps<'li'>) {
447
+ return (
448
+ <li
449
+ data-slot="sidebar-menu-item"
450
+ data-sidebar="menu-item"
451
+ className={cn('group/menu-item relative', className)}
452
+ {...props}
453
+ />
454
+ );
455
+ }
456
+
457
+ const sidebarMenuButtonVariants = cva(
458
+ 'peer/menu-button flex w-full items-center gap-2 overflow-hidden rounded-md p-2 text-left text-sm outline-hidden ring-sidebar-ring transition-[width,height,padding] hover:bg-sidebar-accent hover:text-sidebar-accent-foreground focus-visible:ring-2 active:bg-sidebar-accent active:text-sidebar-accent-foreground disabled:pointer-events-none disabled:opacity-50 group-has-data-[sidebar=menu-action]/menu-item:pr-8 aria-disabled:pointer-events-none aria-disabled:opacity-50 data-[active=true]:bg-sidebar-accent data-[active=true]:font-medium data-[active=true]:text-sidebar-accent-foreground data-[state=open]:hover:bg-sidebar-accent data-[state=open]:hover:text-sidebar-accent-foreground group-data-[collapsible=icon]:size-8! group-data-[collapsible=icon]:p-2! [&>span:last-child]:truncate [&>svg]:size-4 [&>svg]:shrink-0',
459
+ {
460
+ variants: {
461
+ variant: {
462
+ default: 'hover:bg-sidebar-accent hover:text-sidebar-accent-foreground',
463
+ outline:
464
+ 'bg-background shadow-[0_0_0_1px_hsl(var(--sidebar-border))] hover:bg-sidebar-accent hover:text-sidebar-accent-foreground hover:shadow-[0_0_0_1px_hsl(var(--sidebar-accent))]',
465
+ },
466
+ size: {
467
+ default: 'h-8 text-sm',
468
+ sm: 'h-7 text-xs',
469
+ lg: 'h-12 text-sm group-data-[collapsible=icon]:p-0!',
470
+ },
471
+ },
472
+ defaultVariants: {
473
+ variant: 'default',
474
+ size: 'default',
475
+ },
476
+ },
477
+ );
478
+
479
+ function SidebarMenuButton({
480
+ asChild = false,
481
+ isActive = false,
482
+ variant = 'default',
483
+ size = 'default',
484
+ tooltip,
485
+ className,
486
+ ...props
487
+ }: React.ComponentProps<'button'> & {
488
+ asChild?: boolean;
489
+ isActive?: boolean;
490
+ tooltip?: string | React.ComponentProps<typeof TooltipContent>;
491
+ } & VariantProps<typeof sidebarMenuButtonVariants>) {
492
+ const Comp = asChild ? Slot : 'button';
493
+ const { isMobile, state } = useSidebar();
494
+
495
+ const button = (
496
+ <Comp
497
+ data-slot="sidebar-menu-button"
498
+ data-sidebar="menu-button"
499
+ data-size={size}
500
+ data-active={isActive}
501
+ className={cn(sidebarMenuButtonVariants({ variant, size }), className)}
502
+ {...props}
503
+ />
504
+ );
505
+
506
+ if (!tooltip) {
507
+ return button;
508
+ }
509
+
510
+ if (typeof tooltip === 'string') {
511
+ tooltip = {
512
+ children: tooltip,
513
+ };
514
+ }
515
+
516
+ return (
517
+ <Tooltip>
518
+ <TooltipTrigger asChild>{button}</TooltipTrigger>
519
+ <TooltipContent
520
+ side="right"
521
+ align="center"
522
+ hidden={state !== 'collapsed' || isMobile}
523
+ {...tooltip}
524
+ />
525
+ </Tooltip>
526
+ );
527
+ }
528
+
529
+ function SidebarMenuAction({
530
+ className,
531
+ asChild = false,
532
+ showOnHover = false,
533
+ ...props
534
+ }: React.ComponentProps<'button'> & {
535
+ asChild?: boolean;
536
+ showOnHover?: boolean;
537
+ }) {
538
+ const Comp = asChild ? Slot : 'button';
539
+
540
+ return (
541
+ <Comp
542
+ data-slot="sidebar-menu-action"
543
+ data-sidebar="menu-action"
544
+ className={cn(
545
+ 'text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground peer-hover/menu-button:text-sidebar-accent-foreground absolute top-1.5 right-1 flex aspect-square w-5 items-center justify-center rounded-md p-0 outline-hidden transition-transform focus-visible:ring-2 [&>svg]:size-4 [&>svg]:shrink-0',
546
+ // Increases the hit area of the button on mobile.
547
+ 'after:absolute after:-inset-2 md:after:hidden',
548
+ 'peer-data-[size=sm]/menu-button:top-1',
549
+ 'peer-data-[size=default]/menu-button:top-1.5',
550
+ 'peer-data-[size=lg]/menu-button:top-2.5',
551
+ 'group-data-[collapsible=icon]:hidden',
552
+ showOnHover &&
553
+ 'peer-data-[active=true]/menu-button:text-sidebar-accent-foreground group-focus-within/menu-item:opacity-100 group-hover/menu-item:opacity-100 data-[state=open]:opacity-100 md:opacity-0',
554
+ className,
555
+ )}
556
+ {...props}
557
+ />
558
+ );
559
+ }
560
+
561
+ function SidebarMenuBadge({ className, ...props }: React.ComponentProps<'div'>) {
562
+ return (
563
+ <div
564
+ data-slot="sidebar-menu-badge"
565
+ data-sidebar="menu-badge"
566
+ className={cn(
567
+ 'text-sidebar-foreground pointer-events-none absolute right-1 flex h-5 min-w-5 items-center justify-center rounded-md px-1 text-xs font-medium tabular-nums select-none',
568
+ 'peer-hover/menu-button:text-sidebar-accent-foreground peer-data-[active=true]/menu-button:text-sidebar-accent-foreground',
569
+ 'peer-data-[size=sm]/menu-button:top-1',
570
+ 'peer-data-[size=default]/menu-button:top-1.5',
571
+ 'peer-data-[size=lg]/menu-button:top-2.5',
572
+ 'group-data-[collapsible=icon]:hidden',
573
+ className,
574
+ )}
575
+ {...props}
576
+ />
577
+ );
578
+ }
579
+
580
+ function SidebarMenuSkeleton({
581
+ className,
582
+ showIcon = false,
583
+ ...props
584
+ }: React.ComponentProps<'div'> & {
585
+ showIcon?: boolean;
586
+ }) {
587
+ // Random width between 50 to 90%.
588
+ const width = React.useMemo(() => {
589
+ return `${Math.floor(Math.random() * 40) + 50}%`;
590
+ }, []);
591
+
592
+ return (
593
+ <div
594
+ data-slot="sidebar-menu-skeleton"
595
+ data-sidebar="menu-skeleton"
596
+ className={cn('flex h-8 items-center gap-2 rounded-md px-2', className)}
597
+ {...props}
598
+ >
599
+ {showIcon && <Skeleton className="size-4 rounded-md" data-sidebar="menu-skeleton-icon" />}
600
+ <Skeleton
601
+ className="h-4 max-w-(--skeleton-width) flex-1"
602
+ data-sidebar="menu-skeleton-text"
603
+ style={
604
+ {
605
+ '--skeleton-width': width,
606
+ } as React.CSSProperties
607
+ }
608
+ />
609
+ </div>
610
+ );
611
+ }
612
+
613
+ function SidebarMenuSub({ className, ...props }: React.ComponentProps<'ul'>) {
614
+ return (
615
+ <ul
616
+ data-slot="sidebar-menu-sub"
617
+ data-sidebar="menu-sub"
618
+ className={cn(
619
+ 'border-sidebar-border mx-3.5 flex min-w-0 translate-x-px flex-col gap-1 border-l px-2.5 py-0.5',
620
+ 'group-data-[collapsible=icon]:hidden',
621
+ className,
622
+ )}
623
+ {...props}
624
+ />
625
+ );
626
+ }
627
+
628
+ function SidebarMenuSubItem({ className, ...props }: React.ComponentProps<'li'>) {
629
+ return (
630
+ <li
631
+ data-slot="sidebar-menu-sub-item"
632
+ data-sidebar="menu-sub-item"
633
+ className={cn('group/menu-sub-item relative', className)}
634
+ {...props}
635
+ />
636
+ );
637
+ }
638
+
639
+ function SidebarMenuSubButton({
640
+ asChild = false,
641
+ size = 'md',
642
+ isActive = false,
643
+ className,
644
+ ...props
645
+ }: React.ComponentProps<'a'> & {
646
+ asChild?: boolean;
647
+ size?: 'sm' | 'md';
648
+ isActive?: boolean;
649
+ }) {
650
+ const Comp = asChild ? Slot : 'a';
651
+
652
+ return (
653
+ <Comp
654
+ data-slot="sidebar-menu-sub-button"
655
+ data-sidebar="menu-sub-button"
656
+ data-size={size}
657
+ data-active={isActive}
658
+ className={cn(
659
+ 'text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground active:bg-sidebar-accent active:text-sidebar-accent-foreground [&>svg]:text-sidebar-accent-foreground flex h-7 min-w-0 -translate-x-px items-center gap-2 overflow-hidden rounded-md px-2 outline-hidden focus-visible:ring-2 disabled:pointer-events-none disabled:opacity-50 aria-disabled:pointer-events-none aria-disabled:opacity-50 [&>span:last-child]:truncate [&>svg]:size-4 [&>svg]:shrink-0',
660
+ 'data-[active=true]:bg-sidebar-accent data-[active=true]:text-sidebar-accent-foreground',
661
+ size === 'sm' && 'text-xs',
662
+ size === 'md' && 'text-sm',
663
+ 'group-data-[collapsible=icon]:hidden',
664
+ className,
665
+ )}
666
+ {...props}
667
+ />
668
+ );
669
+ }
670
+
671
+ export {
672
+ Sidebar,
673
+ SidebarContent,
674
+ SidebarFooter,
675
+ SidebarGroup,
676
+ SidebarGroupAction,
677
+ SidebarGroupContent,
678
+ SidebarGroupLabel,
679
+ SidebarHeader,
680
+ SidebarInput,
681
+ SidebarInset,
682
+ SidebarMenu,
683
+ SidebarMenuAction,
684
+ SidebarMenuBadge,
685
+ SidebarMenuButton,
686
+ SidebarMenuItem,
687
+ SidebarMenuSkeleton,
688
+ SidebarMenuSub,
689
+ SidebarMenuSubButton,
690
+ SidebarMenuSubItem,
691
+ SidebarProvider,
692
+ SidebarRail,
693
+ SidebarSeparator,
694
+ SidebarTrigger,
695
+ useSidebar,
696
+ };