@vendure/dashboard 3.2.0 → 3.2.2

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