@spaceinvoices/react-ui 0.1.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 (352) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +340 -0
  3. package/cli/dist/index.js +922 -0
  4. package/package.json +87 -0
  5. package/registry.json +600 -0
  6. package/spaceinvoices.schema.json +47 -0
  7. package/src/app.tsx +25 -0
  8. package/src/common/autocomplete.tsx +135 -0
  9. package/src/components/activities/activity-timeline.tsx +160 -0
  10. package/src/components/activities/index.ts +1 -0
  11. package/src/components/activities/locales/de.ts +30 -0
  12. package/src/components/activities/locales/sl.ts +30 -0
  13. package/src/components/advance-invoices/advance-invoices.hooks.ts +75 -0
  14. package/src/components/advance-invoices/create/create-advance-invoice-form.tsx +702 -0
  15. package/src/components/advance-invoices/create/locales/de.ts +29 -0
  16. package/src/components/advance-invoices/create/locales/sl.ts +25 -0
  17. package/src/components/advance-invoices/create/prepare-advance-invoice-submission.ts +74 -0
  18. package/src/components/advance-invoices/index.ts +5 -0
  19. package/src/components/advance-invoices/list/index.ts +3 -0
  20. package/src/components/advance-invoices/list/list-row-actions.tsx +119 -0
  21. package/src/components/advance-invoices/list/list-table.tsx +178 -0
  22. package/src/components/advance-invoices/list/locales/de.ts +32 -0
  23. package/src/components/advance-invoices/list/locales/sl.ts +32 -0
  24. package/src/components/advance-invoices/list/use-advance-invoice-download.ts +63 -0
  25. package/src/components/button-loader.tsx +11 -0
  26. package/src/components/combobox.tsx +96 -0
  27. package/src/components/company-registry/company-registry-autocomplete.tsx +151 -0
  28. package/src/components/company-registry/company-registry.hooks.ts +67 -0
  29. package/src/components/company-registry/index.ts +7 -0
  30. package/src/components/credit-notes/create/create-credit-note-form.tsx +332 -0
  31. package/src/components/credit-notes/create/index.ts +1 -0
  32. package/src/components/credit-notes/create/locales/de.ts +69 -0
  33. package/src/components/credit-notes/create/locales/sl.ts +67 -0
  34. package/src/components/credit-notes/credit-notes.hooks.ts +22 -0
  35. package/src/components/credit-notes/index.ts +10 -0
  36. package/src/components/credit-notes/list/index.ts +3 -0
  37. package/src/components/credit-notes/list/list-row-actions.tsx +116 -0
  38. package/src/components/credit-notes/list/list-table.tsx +183 -0
  39. package/src/components/credit-notes/list/locales/de.ts +33 -0
  40. package/src/components/credit-notes/list/locales/sl.ts +33 -0
  41. package/src/components/credit-notes/list/use-credit-note-download.ts +65 -0
  42. package/src/components/customers/create-customer-form/create-customer-form.tsx +134 -0
  43. package/src/components/customers/create-customer-form/locales/de.ts +20 -0
  44. package/src/components/customers/create-customer-form/locales/sl.ts +20 -0
  45. package/src/components/customers/customer-autocomplete.tsx +173 -0
  46. package/src/components/customers/customer-combobox.tsx +130 -0
  47. package/src/components/customers/customer-list-table/customer-list-row-actions.tsx +48 -0
  48. package/src/components/customers/customer-list-table/customer-list-table.tsx +124 -0
  49. package/src/components/customers/customer-list-table/index.ts +2 -0
  50. package/src/components/customers/customer-list-table/locales/de.ts +16 -0
  51. package/src/components/customers/customer-list-table/locales/sl.ts +16 -0
  52. package/src/components/customers/customers.hooks.test.ts +348 -0
  53. package/src/components/customers/customers.hooks.ts +57 -0
  54. package/src/components/customers/index.ts +5 -0
  55. package/src/components/dashboard/chart-empty-state.tsx +29 -0
  56. package/src/components/dashboard/collection-rate-card/collection-rate-card.tsx +80 -0
  57. package/src/components/dashboard/collection-rate-card/index.ts +4 -0
  58. package/src/components/dashboard/collection-rate-card/locales/sl.ts +3 -0
  59. package/src/components/dashboard/collection-rate-card/use-collection-rate.ts +74 -0
  60. package/src/components/dashboard/index.ts +54 -0
  61. package/src/components/dashboard/invoice-status-chart/index.ts +4 -0
  62. package/src/components/dashboard/invoice-status-chart/invoice-status-chart.tsx +130 -0
  63. package/src/components/dashboard/invoice-status-chart/locales/sl.ts +9 -0
  64. package/src/components/dashboard/invoice-status-chart/use-invoice-status.ts +105 -0
  65. package/src/components/dashboard/loading-card.tsx +19 -0
  66. package/src/components/dashboard/payment-methods-chart/index.ts +4 -0
  67. package/src/components/dashboard/payment-methods-chart/locales/sl.ts +12 -0
  68. package/src/components/dashboard/payment-methods-chart/payment-methods-chart.tsx +152 -0
  69. package/src/components/dashboard/payment-methods-chart/use-payment-methods.ts +50 -0
  70. package/src/components/dashboard/payment-trend-chart/index.ts +4 -0
  71. package/src/components/dashboard/payment-trend-chart/locales/sl.ts +5 -0
  72. package/src/components/dashboard/payment-trend-chart/payment-trend-chart.tsx +137 -0
  73. package/src/components/dashboard/payment-trend-chart/use-payment-trend.ts +92 -0
  74. package/src/components/dashboard/revenue-card.tsx +49 -0
  75. package/src/components/dashboard/revenue-trend-chart/index.ts +4 -0
  76. package/src/components/dashboard/revenue-trend-chart/locales/sl.ts +5 -0
  77. package/src/components/dashboard/revenue-trend-chart/revenue-trend-chart.tsx +137 -0
  78. package/src/components/dashboard/revenue-trend-chart/use-revenue-trend.ts +93 -0
  79. package/src/components/dashboard/shared/index.ts +5 -0
  80. package/src/components/dashboard/shared/use-revenue-data.ts +160 -0
  81. package/src/components/dashboard/shared/use-stats-counts.ts +89 -0
  82. package/src/components/dashboard/shared/use-stats-query.ts +38 -0
  83. package/src/components/dashboard/stat-card.tsx +41 -0
  84. package/src/components/dashboard/tax-collected-card/index.ts +2 -0
  85. package/src/components/dashboard/tax-collected-card/tax-collected-card.tsx +77 -0
  86. package/src/components/dashboard/tax-collected-card/use-tax-collected.ts +145 -0
  87. package/src/components/dashboard/top-customers-chart/index.ts +4 -0
  88. package/src/components/dashboard/top-customers-chart/locales/sl.ts +5 -0
  89. package/src/components/dashboard/top-customers-chart/top-customers-chart.tsx +130 -0
  90. package/src/components/dashboard/top-customers-chart/use-top-customers.ts +72 -0
  91. package/src/components/documents/create/document-add-item-form.tsx +379 -0
  92. package/src/components/documents/create/document-add-item-tax-rate-field.tsx +120 -0
  93. package/src/components/documents/create/document-details-section.tsx +597 -0
  94. package/src/components/documents/create/document-items-section.tsx +133 -0
  95. package/src/components/documents/create/document-recipient-section.tsx +101 -0
  96. package/src/components/documents/create/form-types.ts +36 -0
  97. package/src/components/documents/create/index.ts +9 -0
  98. package/src/components/documents/create/live-preview.tsx +235 -0
  99. package/src/components/documents/create/mark-as-paid-section.tsx +82 -0
  100. package/src/components/documents/create/prepare-document-submission.test.ts +132 -0
  101. package/src/components/documents/create/prepare-document-submission.ts +187 -0
  102. package/src/components/documents/create/prepare-preview-data.test.ts +155 -0
  103. package/src/components/documents/create/prepare-preview-data.ts +16 -0
  104. package/src/components/documents/create/smart-code-insert-button.tsx +139 -0
  105. package/src/components/documents/create/use-document-customer-form.ts +161 -0
  106. package/src/components/documents/document-preview.tsx +13 -0
  107. package/src/components/documents/documents.hooks.ts +146 -0
  108. package/src/components/documents/index.ts +23 -0
  109. package/src/components/documents/shared/document-preview-display.tsx +172 -0
  110. package/src/components/documents/shared/index.ts +3 -0
  111. package/src/components/documents/shared/scaled-document-preview.tsx +70 -0
  112. package/src/components/documents/shared/use-a4-scaling.ts +62 -0
  113. package/src/components/documents/types.ts +61 -0
  114. package/src/components/documents/view/document-actions-bar.tsx +328 -0
  115. package/src/components/documents/view/document-details-card.tsx +179 -0
  116. package/src/components/documents/view/document-payments-list.tsx +256 -0
  117. package/src/components/documents/view/index.ts +4 -0
  118. package/src/components/documents/view/locales/de.ts +85 -0
  119. package/src/components/documents/view/locales/sl.ts +84 -0
  120. package/src/components/documents/view/use-document-download.ts +125 -0
  121. package/src/components/entities/create-entity-form.tsx +105 -0
  122. package/src/components/entities/entities.hooks.ts +50 -0
  123. package/src/components/entities/entity-settings-form/email-template-variables-info.tsx +103 -0
  124. package/src/components/entities/entity-settings-form/entity-settings-form.tsx +1326 -0
  125. package/src/components/entities/entity-settings-form/image-upload-with-crop.tsx +222 -0
  126. package/src/components/entities/entity-settings-form/index.ts +2 -0
  127. package/src/components/entities/entity-settings-form/input-with-preview.tsx +190 -0
  128. package/src/components/entities/entity-settings-form/locales/de.ts +192 -0
  129. package/src/components/entities/entity-settings-form/locales/sl.ts +188 -0
  130. package/src/components/entities/furs-settings-form/furs-settings-form.tsx +410 -0
  131. package/src/components/entities/furs-settings-form/furs-settings.hooks.ts +320 -0
  132. package/src/components/entities/furs-settings-form/index.ts +3 -0
  133. package/src/components/entities/furs-settings-form/locales/de.ts +233 -0
  134. package/src/components/entities/furs-settings-form/locales/en.ts +194 -0
  135. package/src/components/entities/furs-settings-form/locales/sl.ts +196 -0
  136. package/src/components/entities/furs-settings-form/sections/certificate-settings-section.tsx +242 -0
  137. package/src/components/entities/furs-settings-form/sections/enable-fiscalization-section.tsx +139 -0
  138. package/src/components/entities/furs-settings-form/sections/general-settings-section.tsx +252 -0
  139. package/src/components/entities/furs-settings-form/sections/premises-management-section.tsx +370 -0
  140. package/src/components/entities/furs-settings-form/sections/register-premise-dialog.tsx +420 -0
  141. package/src/components/entities/keys.ts +2 -0
  142. package/src/components/entities/settings/branding-settings-form.tsx +274 -0
  143. package/src/components/entities/settings/company-settings-form.tsx +256 -0
  144. package/src/components/entities/settings/defaults-settings-form.tsx +501 -0
  145. package/src/components/entities/settings/email-settings-form.tsx +288 -0
  146. package/src/components/entities/settings/eslog-settings-form.tsx +113 -0
  147. package/src/components/entities/settings/index.ts +8 -0
  148. package/src/components/entities/settings/number-format-settings-form.tsx +244 -0
  149. package/src/components/entities/settings/pdf-template-selector/demo-invoice-data.ts +164 -0
  150. package/src/components/entities/settings/pdf-template-selector/index.ts +2 -0
  151. package/src/components/entities/settings/pdf-template-selector/locales/de.ts +18 -0
  152. package/src/components/entities/settings/pdf-template-selector/locales/sl.ts +18 -0
  153. package/src/components/entities/settings/pdf-template-selector/pdf-template-cards.tsx +49 -0
  154. package/src/components/entities/settings/settings-footer.tsx +16 -0
  155. package/src/components/entities/settings/tax-rules-settings-form.tsx +346 -0
  156. package/src/components/estimates/create/create-estimate-form.tsx +384 -0
  157. package/src/components/estimates/create/locales/de.ts +64 -0
  158. package/src/components/estimates/create/locales/sl.ts +63 -0
  159. package/src/components/estimates/create/prepare-estimate-submission.ts +39 -0
  160. package/src/components/estimates/create/use-estimate-customer-form.ts +5 -0
  161. package/src/components/estimates/estimates.hooks.ts +15 -0
  162. package/src/components/estimates/index.ts +6 -0
  163. package/src/components/estimates/list/index.ts +3 -0
  164. package/src/components/estimates/list/list-row-actions.tsx +103 -0
  165. package/src/components/estimates/list/list-table.tsx +171 -0
  166. package/src/components/estimates/list/locales/de.ts +26 -0
  167. package/src/components/estimates/list/locales/sl.ts +26 -0
  168. package/src/components/estimates/list/use-estimate-download.ts +63 -0
  169. package/src/components/export/document-export-form.tsx +288 -0
  170. package/src/components/export/index.ts +2 -0
  171. package/src/components/form/form-input.tsx +89 -0
  172. package/src/components/form/index.ts +1 -0
  173. package/src/components/invoices/create/create-invoice-form.tsx +852 -0
  174. package/src/components/invoices/create/eslog-validation.test.ts +242 -0
  175. package/src/components/invoices/create/eslog-validation.ts +208 -0
  176. package/src/components/invoices/create/locales/de.ts +118 -0
  177. package/src/components/invoices/create/locales/sl.ts +114 -0
  178. package/src/components/invoices/create/prepare-invoice-submission.test.ts +777 -0
  179. package/src/components/invoices/create/prepare-invoice-submission.ts +79 -0
  180. package/src/components/invoices/create/use-invoice-customer-form.ts +5 -0
  181. package/src/components/invoices/index.ts +9 -0
  182. package/src/components/invoices/invoices-furs.hooks.ts +28 -0
  183. package/src/components/invoices/invoices.hooks.ts +110 -0
  184. package/src/components/invoices/list/index.ts +3 -0
  185. package/src/components/invoices/list/list-row-actions.tsx +132 -0
  186. package/src/components/invoices/list/list-table.tsx +165 -0
  187. package/src/components/invoices/list/locales/de.ts +33 -0
  188. package/src/components/invoices/list/locales/sl.ts +33 -0
  189. package/src/components/invoices/list/use-invoice-download.ts +62 -0
  190. package/src/components/invoices/send-email-dialog/index.ts +1 -0
  191. package/src/components/invoices/send-email-dialog/locales/de.ts +18 -0
  192. package/src/components/invoices/send-email-dialog/locales/sl.ts +17 -0
  193. package/src/components/invoices/send-email-dialog/send-email-dialog.tsx +289 -0
  194. package/src/components/invoices/send-email-dialog.tsx +2 -0
  195. package/src/components/invoices/shared/index.ts +2 -0
  196. package/src/components/invoices/shared/scaled-document-preview.tsx +32 -0
  197. package/src/components/invoices/shared/use-a4-scaling.tsx +39 -0
  198. package/src/components/invoices/view/eslog-info-display.tsx +160 -0
  199. package/src/components/invoices/view/furs-info-display.tsx +213 -0
  200. package/src/components/items/create-item-form/create-item-form.tsx +155 -0
  201. package/src/components/items/create-item-form/locales/de.ts +14 -0
  202. package/src/components/items/create-item-form/locales/en.ts +9 -0
  203. package/src/components/items/create-item-form/locales/sl.ts +14 -0
  204. package/src/components/items/item-combobox.tsx +147 -0
  205. package/src/components/items/item-list-table/item-list-header.tsx +33 -0
  206. package/src/components/items/item-list-table/item-list-row-actions.tsx +48 -0
  207. package/src/components/items/item-list-table/item-list-row.tsx +32 -0
  208. package/src/components/items/item-list-table/item-list-table.tsx +76 -0
  209. package/src/components/items/item-list-table/locales/de.ts +10 -0
  210. package/src/components/items/item-list-table/locales/en.ts +10 -0
  211. package/src/components/items/item-list-table/locales/sl.ts +10 -0
  212. package/src/components/items/items.hooks.ts +63 -0
  213. package/src/components/loading-spinner.tsx +24 -0
  214. package/src/components/payments/create-payment-form/create-payment-form.tsx +222 -0
  215. package/src/components/payments/create-payment-form/locales/de.ts +20 -0
  216. package/src/components/payments/create-payment-form/locales/sl.ts +20 -0
  217. package/src/components/payments/edit-payment-form/edit-payment-form.tsx +230 -0
  218. package/src/components/payments/edit-payment-form/index.ts +1 -0
  219. package/src/components/payments/edit-payment-form/locales/de.ts +20 -0
  220. package/src/components/payments/edit-payment-form/locales/sl.ts +20 -0
  221. package/src/components/payments/index.ts +4 -0
  222. package/src/components/payments/list/index.ts +2 -0
  223. package/src/components/payments/list/list-row-actions.tsx +98 -0
  224. package/src/components/payments/list/list-table.tsx +186 -0
  225. package/src/components/payments/list/locales/de.ts +19 -0
  226. package/src/components/payments/list/locales/sl.ts +19 -0
  227. package/src/components/payments/payments.hooks.ts +15 -0
  228. package/src/components/request-logs/index.ts +3 -0
  229. package/src/components/request-logs/request-log-detail.tsx +242 -0
  230. package/src/components/request-logs/request-log-list-table.tsx +266 -0
  231. package/src/components/request-logs/request-logs-page.tsx +10 -0
  232. package/src/components/table/README.md +410 -0
  233. package/src/components/table/data-table.tsx +251 -0
  234. package/src/components/table/date-cell.tsx +35 -0
  235. package/src/components/table/filter-bar.tsx +114 -0
  236. package/src/components/table/filter-panel.tsx +407 -0
  237. package/src/components/table/hooks/use-table-fetch.ts +17 -0
  238. package/src/components/table/hooks/use-table-query.ts +36 -0
  239. package/src/components/table/hooks/use-table-state.ts +293 -0
  240. package/src/components/table/index.ts +35 -0
  241. package/src/components/table/search-input.tsx +85 -0
  242. package/src/components/table/sortable-header.tsx +56 -0
  243. package/src/components/table/table-empty-state.tsx +40 -0
  244. package/src/components/table/table-no-results.tsx +41 -0
  245. package/src/components/table/table-pagination.tsx +42 -0
  246. package/src/components/table/table-skeleton.tsx +54 -0
  247. package/src/components/table/types.ts +136 -0
  248. package/src/components/tax-reports/index.ts +1 -0
  249. package/src/components/tax-reports/kir-export-form.tsx +172 -0
  250. package/src/components/taxes/create-tax-form/create-tax-form.tsx +112 -0
  251. package/src/components/taxes/create-tax-form/locales/de.ts +8 -0
  252. package/src/components/taxes/create-tax-form/locales/en.ts +7 -0
  253. package/src/components/taxes/create-tax-form/locales/sl.ts +8 -0
  254. package/src/components/taxes/tax-list-table/locales/de.ts +11 -0
  255. package/src/components/taxes/tax-list-table/locales/en.ts +10 -0
  256. package/src/components/taxes/tax-list-table/locales/sl.ts +11 -0
  257. package/src/components/taxes/tax-list-table/tax-list-header.tsx +29 -0
  258. package/src/components/taxes/tax-list-table/tax-list-row-actions.tsx +43 -0
  259. package/src/components/taxes/tax-list-table/tax-list-row.tsx +46 -0
  260. package/src/components/taxes/tax-list-table/tax-list-table.tsx +59 -0
  261. package/src/components/taxes/taxes.hooks.ts +35 -0
  262. package/src/components/ui/alert-dialog.tsx +61 -0
  263. package/src/components/ui/alert.tsx +72 -0
  264. package/src/components/ui/badge.tsx +48 -0
  265. package/src/components/ui/breadcrumb.tsx +132 -0
  266. package/src/components/ui/button.tsx +61 -0
  267. package/src/components/ui/calendar.tsx +213 -0
  268. package/src/components/ui/card.tsx +94 -0
  269. package/src/components/ui/chart.tsx +380 -0
  270. package/src/components/ui/checkbox.tsx +27 -0
  271. package/src/components/ui/collapsible.tsx +56 -0
  272. package/src/components/ui/command.tsx +187 -0
  273. package/src/components/ui/dialog.tsx +187 -0
  274. package/src/components/ui/drawer.tsx +123 -0
  275. package/src/components/ui/dropdown-menu.tsx +291 -0
  276. package/src/components/ui/form.tsx +166 -0
  277. package/src/components/ui/input-group.tsx +149 -0
  278. package/src/components/ui/input.tsx +20 -0
  279. package/src/components/ui/label.tsx +18 -0
  280. package/src/components/ui/loading-spinner.tsx +16 -0
  281. package/src/components/ui/popover.tsx +108 -0
  282. package/src/components/ui/radio-group.tsx +37 -0
  283. package/src/components/ui/select.tsx +200 -0
  284. package/src/components/ui/separator.tsx +23 -0
  285. package/src/components/ui/sheet.tsx +145 -0
  286. package/src/components/ui/sidebar.tsx +771 -0
  287. package/src/components/ui/skeleton.tsx +13 -0
  288. package/src/components/ui/sonner.tsx +60 -0
  289. package/src/components/ui/spinner.tsx +10 -0
  290. package/src/components/ui/sticky-form-footer.tsx +55 -0
  291. package/src/components/ui/switch.tsx +30 -0
  292. package/src/components/ui/table.tsx +101 -0
  293. package/src/components/ui/tabs.tsx +80 -0
  294. package/src/components/ui/textarea.tsx +18 -0
  295. package/src/components/ui/tooltip.tsx +89 -0
  296. package/src/components/wl-subscription/index.ts +2 -0
  297. package/src/components/wl-subscription/locked-feature.tsx +173 -0
  298. package/src/components/wl-subscription/upgrade-modal.tsx +209 -0
  299. package/src/frontend.tsx +28 -0
  300. package/src/generate-schemas.ts +265 -0
  301. package/src/generated/schemas/advanceinvoice.ts +177 -0
  302. package/src/generated/schemas/creditnote.ts +187 -0
  303. package/src/generated/schemas/customer.ts +29 -0
  304. package/src/generated/schemas/entity.ts +252 -0
  305. package/src/generated/schemas/estimate.ts +159 -0
  306. package/src/generated/schemas/furssettings.ts +25 -0
  307. package/src/generated/schemas/index.ts +24 -0
  308. package/src/generated/schemas/invoice.ts +167 -0
  309. package/src/generated/schemas/item.ts +38 -0
  310. package/src/generated/schemas/payment.ts +44 -0
  311. package/src/generated/schemas/previewadvanceinvoice_body.ts +354 -0
  312. package/src/generated/schemas/previewestimate_body.ts +309 -0
  313. package/src/generated/schemas/registerfursmovablepremise_body.ts +22 -0
  314. package/src/generated/schemas/registerfursrealestatepremise_body.ts +32 -0
  315. package/src/generated/schemas/renderdocument_body.ts +594 -0
  316. package/src/generated/schemas/sendemail_body.ts +26 -0
  317. package/src/generated/schemas/startpdfexport_body.ts +20 -0
  318. package/src/generated/schemas/tax.ts +48 -0
  319. package/src/generated/schemas/uploadfile_body.ts +23 -0
  320. package/src/generated/schemas/uploadfurscertificate_body.ts +20 -0
  321. package/src/generated/schemas/userfurssettings.ts +19 -0
  322. package/src/hooks/create-resource-hooks.test.ts +483 -0
  323. package/src/hooks/create-resource-hooks.ts +300 -0
  324. package/src/hooks/use-debounce.ts +12 -0
  325. package/src/hooks/use-duplicate-document.ts +185 -0
  326. package/src/hooks/use-media-query.tsx +19 -0
  327. package/src/hooks/use-mobile.ts +39 -0
  328. package/src/hooks/use-next-document-number.ts +57 -0
  329. package/src/hooks/use-resource-mutation.ts +118 -0
  330. package/src/hooks/use-vies-check.ts +130 -0
  331. package/src/index.css +11 -0
  332. package/src/index.html +13 -0
  333. package/src/index.tsx +12 -0
  334. package/src/lib/auth.ts +4 -0
  335. package/src/lib/browser-cookies.ts +70 -0
  336. package/src/lib/constants.ts +287 -0
  337. package/src/lib/cookies.ts +36 -0
  338. package/src/lib/schemas/advance-invoice.ts +43 -0
  339. package/src/lib/schemas/credit-note.ts +32 -0
  340. package/src/lib/schemas/estimate.ts +31 -0
  341. package/src/lib/schemas/index.ts +18 -0
  342. package/src/lib/schemas/invoice.ts +43 -0
  343. package/src/lib/schemas/shared.ts +79 -0
  344. package/src/lib/translation.ts +38 -0
  345. package/src/lib/utils.ts +6 -0
  346. package/src/providers/entities-context.tsx +41 -0
  347. package/src/providers/entities-provider.tsx +201 -0
  348. package/src/providers/form-footer-context.tsx +72 -0
  349. package/src/providers/sdk-provider.tsx +164 -0
  350. package/src/providers/white-label-provider.tsx +91 -0
  351. package/src/providers/wl-subscription-provider.tsx +277 -0
  352. package/src/utils/string-helpers.ts +111 -0
@@ -0,0 +1,771 @@
1
+ import * as React from "react"
2
+ import { mergeProps } from "@base-ui/react/merge-props"
3
+ import { useRender } from "@base-ui/react/use-render"
4
+ import { cva, type VariantProps } from "class-variance-authority"
5
+
6
+ import { cn } from "@/ui/lib/utils"
7
+ import { Button } from "@/ui/components/ui/button"
8
+ import { Input } from "@/ui/components/ui/input"
9
+ import { Separator } from "@/ui/components/ui/separator"
10
+ import {
11
+ Sheet,
12
+ SheetContent,
13
+ SheetDescription,
14
+ SheetHeader,
15
+ SheetTitle,
16
+ } from "@/ui/components/ui/sheet"
17
+ import { Skeleton } from "@/ui/components/ui/skeleton"
18
+ import {
19
+ Tooltip,
20
+ TooltipContent,
21
+ TooltipTrigger,
22
+ } from "@/ui/components/ui/tooltip"
23
+ import { useIsMobile, useIsSmallScreen } from "@/ui/hooks/use-mobile"
24
+ import { PanelLeftIcon } from "lucide-react"
25
+
26
+ const SIDEBAR_COOKIE_NAME = "sidebar_state"
27
+ const SIDEBAR_COOKIE_MAX_AGE = 60 * 60 * 24 * 7
28
+ const SIDEBAR_WIDTH = "16rem"
29
+ const SIDEBAR_WIDTH_MOBILE = "18rem"
30
+ const SIDEBAR_WIDTH_ICON = "3rem"
31
+ const SIDEBAR_KEYBOARD_SHORTCUT = "b"
32
+
33
+ type SidebarContextProps = {
34
+ state: "expanded" | "collapsed"
35
+ open: boolean
36
+ setOpen: (open: boolean) => void
37
+ openMobile: boolean
38
+ setOpenMobile: (open: boolean) => void
39
+ isMobile: boolean
40
+ toggleSidebar: () => void
41
+ }
42
+
43
+ const SidebarContext = React.createContext<SidebarContextProps | null>(null)
44
+
45
+ function useSidebar() {
46
+ const context = React.useContext(SidebarContext)
47
+ if (!context) {
48
+ throw new Error("useSidebar must be used within a SidebarProvider.")
49
+ }
50
+
51
+ return context
52
+ }
53
+
54
+ function SidebarProvider({
55
+ defaultOpen = true,
56
+ open: openProp,
57
+ onOpenChange: setOpenProp,
58
+ overrideOpen,
59
+ className,
60
+ style,
61
+ children,
62
+ ...props
63
+ }: React.ComponentProps<"div"> & {
64
+ defaultOpen?: boolean
65
+ open?: boolean
66
+ onOpenChange?: (open: boolean) => void
67
+ overrideOpen?: boolean
68
+ }) {
69
+ const isMobile = useIsMobile()
70
+ const isSmallScreen = useIsSmallScreen()
71
+ const [openMobile, setOpenMobile] = React.useState(false)
72
+ const prevIsSmallScreen = React.useRef(isSmallScreen)
73
+
74
+ // This is the internal state of the sidebar.
75
+ // We use openProp and setOpenProp for control from outside the component.
76
+ const [_open, _setOpen] = React.useState(defaultOpen)
77
+ const persistedOpen = openProp ?? _open
78
+ // Use override if provided, otherwise use persisted state
79
+ const open = overrideOpen !== undefined ? overrideOpen : persistedOpen
80
+ const setOpen = React.useCallback(
81
+ (value: boolean | ((value: boolean) => boolean)) => {
82
+ // Don't persist state changes when overridden (e.g., on document create page)
83
+ if (overrideOpen !== undefined) return
84
+
85
+ const openState = typeof value === "function" ? value(persistedOpen) : value
86
+ if (setOpenProp) {
87
+ setOpenProp(openState)
88
+ } else {
89
+ _setOpen(openState)
90
+ }
91
+
92
+ // This sets the cookie to keep the sidebar state.
93
+ document.cookie = `${SIDEBAR_COOKIE_NAME}=${openState}; path=/; max-age=${SIDEBAR_COOKIE_MAX_AGE}`
94
+ },
95
+ [setOpenProp, persistedOpen, overrideOpen]
96
+ )
97
+
98
+ // Helper to toggle the sidebar.
99
+ const toggleSidebar = React.useCallback(() => {
100
+ return isMobile ? setOpenMobile((open) => !open) : setOpen((open) => !open)
101
+ }, [isMobile, setOpen, setOpenMobile])
102
+
103
+ // Adds a keyboard shortcut to toggle the sidebar.
104
+ React.useEffect(() => {
105
+ const handleKeyDown = (event: KeyboardEvent) => {
106
+ if (
107
+ event.key === SIDEBAR_KEYBOARD_SHORTCUT &&
108
+ (event.metaKey || event.ctrlKey)
109
+ ) {
110
+ event.preventDefault()
111
+ toggleSidebar()
112
+ }
113
+ }
114
+
115
+ window.addEventListener("keydown", handleKeyDown)
116
+ return () => window.removeEventListener("keydown", handleKeyDown)
117
+ }, [toggleSidebar])
118
+
119
+ // Auto-collapse sidebar when entering small screen mode (768-1024px)
120
+ React.useEffect(() => {
121
+ if (isSmallScreen && !prevIsSmallScreen.current) {
122
+ _setOpen(false)
123
+ }
124
+ prevIsSmallScreen.current = isSmallScreen
125
+ }, [isSmallScreen])
126
+
127
+ // We add a state so that we can do data-state="expanded" or "collapsed".
128
+ // This makes it easier to style the sidebar with Tailwind classes.
129
+ const state = open ? "expanded" : "collapsed"
130
+
131
+ const contextValue = React.useMemo<SidebarContextProps>(
132
+ () => ({
133
+ state,
134
+ open,
135
+ setOpen,
136
+ isMobile,
137
+ openMobile,
138
+ setOpenMobile,
139
+ toggleSidebar,
140
+ }),
141
+ [state, open, setOpen, isMobile, openMobile, setOpenMobile, toggleSidebar]
142
+ )
143
+
144
+ return (
145
+ <SidebarContext.Provider value={contextValue}>
146
+ <div
147
+ data-slot="sidebar-wrapper"
148
+ style={
149
+ {
150
+ "--sidebar-width": SIDEBAR_WIDTH,
151
+ "--sidebar-width-icon": SIDEBAR_WIDTH_ICON,
152
+ ...style,
153
+ } as React.CSSProperties
154
+ }
155
+ className={cn(
156
+ "group/sidebar-wrapper has-data-[variant=inset]:bg-sidebar flex min-h-svh w-full",
157
+ className
158
+ )}
159
+ {...props}
160
+ >
161
+ {children}
162
+ </div>
163
+ </SidebarContext.Provider>
164
+ )
165
+ }
166
+
167
+ function Sidebar({
168
+ side = "left",
169
+ variant = "sidebar",
170
+ collapsible = "offExamples",
171
+ className,
172
+ children,
173
+ ...props
174
+ }: React.ComponentProps<"div"> & {
175
+ side?: "left" | "right"
176
+ variant?: "sidebar" | "floating" | "inset"
177
+ collapsible?: "offExamples" | "icon" | "none"
178
+ }) {
179
+ const { isMobile, state, openMobile, setOpenMobile } = useSidebar()
180
+
181
+ if (collapsible === "none") {
182
+ return (
183
+ <div
184
+ data-slot="sidebar"
185
+ className={cn(
186
+ "bg-sidebar text-sidebar-foreground flex h-full w-(--sidebar-width) flex-col",
187
+ className
188
+ )}
189
+ {...props}
190
+ >
191
+ {children}
192
+ </div>
193
+ )
194
+ }
195
+
196
+ if (isMobile) {
197
+ return (
198
+ <Sheet open={openMobile} onOpenChange={setOpenMobile} {...props}>
199
+ <SheetContent
200
+ data-sidebar="sidebar"
201
+ data-slot="sidebar"
202
+ data-mobile="true"
203
+ className="bg-sidebar text-sidebar-foreground w-(--sidebar-width) p-0 [&>button]:hidden"
204
+ style={
205
+ {
206
+ "--sidebar-width": SIDEBAR_WIDTH_MOBILE,
207
+ } as React.CSSProperties
208
+ }
209
+ side={side}
210
+ >
211
+ <SheetHeader className="sr-only">
212
+ <SheetTitle>Sidebar</SheetTitle>
213
+ <SheetDescription>Displays the mobile sidebar.</SheetDescription>
214
+ </SheetHeader>
215
+ <div className="flex h-full w-full flex-col">{children}</div>
216
+ </SheetContent>
217
+ </Sheet>
218
+ )
219
+ }
220
+
221
+ return (
222
+ <div
223
+ className="group peer text-sidebar-foreground hidden md:block"
224
+ data-state={state}
225
+ data-collapsible={state === "collapsed" ? collapsible : ""}
226
+ data-variant={variant}
227
+ data-side={side}
228
+ data-slot="sidebar"
229
+ >
230
+ {/* This is what handles the sidebar gap on desktop */}
231
+ <div
232
+ data-slot="sidebar-gap"
233
+ className={cn(
234
+ "transition-[width] duration-200 ease-linear relative w-(--sidebar-width) bg-transparent",
235
+ "group-data-[collapsible=offExamples]:w-0",
236
+ "group-data-[side=right]:rotate-180",
237
+ variant === "floating" || variant === "inset"
238
+ ? "group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4)))]"
239
+ : "group-data-[collapsible=icon]:w-(--sidebar-width-icon)"
240
+ )}
241
+ />
242
+ <div
243
+ data-slot="sidebar-container"
244
+ className={cn(
245
+ "fixed inset-y-0 z-10 hidden h-svh w-(--sidebar-width) transition-[left,right,width] duration-200 ease-linear md:flex",
246
+ side === "left"
247
+ ? "left-0 group-data-[collapsible=offExamples]:left-[calc(var(--sidebar-width)*-1)]"
248
+ : "right-0 group-data-[collapsible=offExamples]:right-[calc(var(--sidebar-width)*-1)]",
249
+ // Adjust the padding for floating and inset variants.
250
+ variant === "floating" || variant === "inset"
251
+ ? "p-2 group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4))+2px)]"
252
+ : "group-data-[collapsible=icon]:w-(--sidebar-width-icon) group-data-[side=left]:border-r group-data-[side=right]:border-l",
253
+ className
254
+ )}
255
+ {...props}
256
+ >
257
+ <div
258
+ data-sidebar="sidebar"
259
+ data-slot="sidebar-inner"
260
+ className="bg-sidebar group-data-[variant=floating]:ring-sidebar-border group-data-[variant=floating]:rounded-lg group-data-[variant=floating]:shadow-sm group-data-[variant=floating]:ring-1 flex size-full flex-col"
261
+ >
262
+ {children}
263
+ </div>
264
+ </div>
265
+ </div>
266
+ )
267
+ }
268
+
269
+ function SidebarTrigger({
270
+ className,
271
+ onClick,
272
+ ...props
273
+ }: React.ComponentProps<typeof Button>) {
274
+ const { toggleSidebar } = useSidebar()
275
+
276
+ return (
277
+ <Button
278
+ data-sidebar="trigger"
279
+ data-slot="sidebar-trigger"
280
+ variant="ghost"
281
+ size="icon-sm"
282
+ className={cn(className)}
283
+ onClick={(event) => {
284
+ onClick?.(event)
285
+ toggleSidebar()
286
+ }}
287
+ {...props}
288
+ >
289
+ <PanelLeftIcon
290
+ />
291
+ <span className="sr-only">Toggle Sidebar</span>
292
+ </Button>
293
+ )
294
+ }
295
+
296
+ function SidebarRail({ className, ...props }: React.ComponentProps<"button">) {
297
+ const { toggleSidebar } = useSidebar()
298
+
299
+ return (
300
+ <button
301
+ data-sidebar="rail"
302
+ data-slot="sidebar-rail"
303
+ aria-label="Toggle Sidebar"
304
+ tabIndex={-1}
305
+ onClick={toggleSidebar}
306
+ title="Toggle Sidebar"
307
+ className={cn(
308
+ "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",
309
+ "in-data-[side=left]:cursor-w-resize in-data-[side=right]:cursor-e-resize",
310
+ "[[data-side=left][data-state=collapsed]_&]:cursor-e-resize [[data-side=right][data-state=collapsed]_&]:cursor-w-resize",
311
+ "hover:group-data-[collapsible=offExamples]:bg-sidebar group-data-[collapsible=offExamples]:translate-x-0 group-data-[collapsible=offExamples]:after:left-full",
312
+ "[[data-side=left][data-collapsible=offExamples]_&]:-right-2",
313
+ "[[data-side=right][data-collapsible=offExamples]_&]:-left-2",
314
+ className
315
+ )}
316
+ {...props}
317
+ />
318
+ )
319
+ }
320
+
321
+ function SidebarInset({ className, ...props }: React.ComponentProps<"main">) {
322
+ return (
323
+ <main
324
+ data-slot="sidebar-inset"
325
+ className={cn(
326
+ "bg-background 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 relative flex w-full flex-1 flex-col",
327
+ className
328
+ )}
329
+ {...props}
330
+ />
331
+ )
332
+ }
333
+
334
+ function SidebarInput({
335
+ className,
336
+ ...props
337
+ }: React.ComponentProps<typeof Input>) {
338
+ return (
339
+ <Input
340
+ data-slot="sidebar-input"
341
+ data-sidebar="input"
342
+ className={cn("bg-background h-8 w-full shadow-none", className)}
343
+ {...props}
344
+ />
345
+ )
346
+ }
347
+
348
+ function SidebarHeader({ className, ...props }: React.ComponentProps<"div">) {
349
+ return (
350
+ <div
351
+ data-slot="sidebar-header"
352
+ data-sidebar="header"
353
+ className={cn("gap-2 p-2 flex flex-col", className)}
354
+ {...props}
355
+ />
356
+ )
357
+ }
358
+
359
+ function SidebarFooter({ className, ...props }: React.ComponentProps<"div">) {
360
+ return (
361
+ <div
362
+ data-slot="sidebar-footer"
363
+ data-sidebar="footer"
364
+ className={cn("gap-2 p-2 flex flex-col", className)}
365
+ {...props}
366
+ />
367
+ )
368
+ }
369
+
370
+ function SidebarSeparator({
371
+ className,
372
+ ...props
373
+ }: React.ComponentProps<typeof Separator>) {
374
+ return (
375
+ <Separator
376
+ data-slot="sidebar-separator"
377
+ data-sidebar="separator"
378
+ className={cn("bg-sidebar-border mx-2 w-auto", className)}
379
+ {...props}
380
+ />
381
+ )
382
+ }
383
+
384
+ function SidebarContent({ className, ...props }: React.ComponentProps<"div">) {
385
+ return (
386
+ <div
387
+ data-slot="sidebar-content"
388
+ data-sidebar="content"
389
+ className={cn(
390
+ "no-scrollbar gap-2 flex min-h-0 flex-1 flex-col overflow-auto group-data-[collapsible=icon]:overflow-hidden",
391
+ className
392
+ )}
393
+ {...props}
394
+ />
395
+ )
396
+ }
397
+
398
+ function SidebarGroup({ className, ...props }: React.ComponentProps<"div">) {
399
+ return (
400
+ <div
401
+ data-slot="sidebar-group"
402
+ data-sidebar="group"
403
+ className={cn(
404
+ "p-2 relative flex w-full min-w-0 flex-col",
405
+ className
406
+ )}
407
+ {...props}
408
+ />
409
+ )
410
+ }
411
+
412
+ function SidebarGroupLabel({
413
+ className,
414
+ render,
415
+ asChild,
416
+ children,
417
+ ...props
418
+ }: useRender.ComponentProps<"div"> & React.ComponentProps<"div"> & { asChild?: boolean }) {
419
+ // Convert asChild to render prop for Base UI compatibility
420
+ const renderProp = asChild && React.isValidElement(children) ? children : render
421
+
422
+ return useRender({
423
+ defaultTagName: "div",
424
+ props: mergeProps<"div">(
425
+ {
426
+ className: cn(
427
+ "text-sidebar-foreground/70 ring-sidebar-ring h-8 rounded-md px-2 text-xs font-medium transition-[margin,opacity] duration-200 ease-linear group-data-[collapsible=icon]:-mt-8 group-data-[collapsible=icon]:opacity-0 focus-visible:ring-2 [&>svg]:size-4 flex shrink-0 items-center outline-hidden [&>svg]:shrink-0",
428
+ className
429
+ ),
430
+ children: asChild ? undefined : children,
431
+ },
432
+ props
433
+ ),
434
+ render: renderProp,
435
+ state: {
436
+ slot: "sidebar-group-label",
437
+ sidebar: "group-label",
438
+ },
439
+ })
440
+ }
441
+
442
+ function SidebarGroupAction({
443
+ className,
444
+ render,
445
+ asChild,
446
+ children,
447
+ ...props
448
+ }: useRender.ComponentProps<"button"> & React.ComponentProps<"button"> & { asChild?: boolean }) {
449
+ // Convert asChild to render prop for Base UI compatibility
450
+ const renderProp = asChild && React.isValidElement(children) ? children : render
451
+
452
+ return useRender({
453
+ defaultTagName: "button",
454
+ props: mergeProps<"button">(
455
+ {
456
+ className: cn(
457
+ "text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground absolute top-3.5 right-3 w-5 rounded-md p-0 focus-visible:ring-2 [&>svg]:size-4 flex aspect-square items-center justify-center outline-hidden transition-transform [&>svg]:shrink-0 after:absolute after:-inset-2 md:after:hidden group-data-[collapsible=icon]:hidden",
458
+ className
459
+ ),
460
+ children: asChild ? undefined : children,
461
+ },
462
+ props
463
+ ),
464
+ render: renderProp,
465
+ state: {
466
+ slot: "sidebar-group-action",
467
+ sidebar: "group-action",
468
+ },
469
+ })
470
+ }
471
+
472
+ function SidebarGroupContent({
473
+ className,
474
+ ...props
475
+ }: React.ComponentProps<"div">) {
476
+ return (
477
+ <div
478
+ data-slot="sidebar-group-content"
479
+ data-sidebar="group-content"
480
+ className={cn("text-sm w-full", className)}
481
+ {...props}
482
+ />
483
+ )
484
+ }
485
+
486
+ function SidebarMenu({ className, ...props }: React.ComponentProps<"ul">) {
487
+ return (
488
+ <ul
489
+ data-slot="sidebar-menu"
490
+ data-sidebar="menu"
491
+ className={cn("gap-1 flex w-full min-w-0 flex-col", className)}
492
+ {...props}
493
+ />
494
+ )
495
+ }
496
+
497
+ function SidebarMenuItem({ className, ...props }: React.ComponentProps<"li">) {
498
+ return (
499
+ <li
500
+ data-slot="sidebar-menu-item"
501
+ data-sidebar="menu-item"
502
+ className={cn("group/menu-item relative", className)}
503
+ {...props}
504
+ />
505
+ )
506
+ }
507
+
508
+ const sidebarMenuButtonVariants = cva(
509
+ "ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground active:bg-sidebar-accent active:text-sidebar-accent-foreground data-active:bg-primary data-active:text-primary-foreground data-open:hover:bg-sidebar-accent data-open:hover:text-sidebar-accent-foreground gap-2 rounded-md p-2 text-left text-sm transition-[width,height,padding] group-has-data-[sidebar=menu-action]/menu-item:pr-8 group-data-[collapsible=icon]:size-8! group-data-[collapsible=icon]:p-2! focus-visible:ring-2 data-active:font-medium peer/menu-button flex w-full items-center overflow-hidden outline-hidden group/menu-button 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",
510
+ {
511
+ variants: {
512
+ variant: {
513
+ default: "hover:bg-sidebar-accent hover:text-sidebar-accent-foreground",
514
+ outline: "bg-background hover:bg-sidebar-accent hover:text-sidebar-accent-foreground shadow-[0_0_0_1px_hsl(var(--sidebar-border))] hover:shadow-[0_0_0_1px_hsl(var(--sidebar-accent))]",
515
+ },
516
+ size: {
517
+ default: "h-8 text-sm",
518
+ sm: "h-7 text-xs",
519
+ lg: "h-12 text-sm group-data-[collapsible=icon]:p-0!",
520
+ },
521
+ },
522
+ defaultVariants: {
523
+ variant: "default",
524
+ size: "default",
525
+ },
526
+ }
527
+ )
528
+
529
+ function SidebarMenuButton({
530
+ render,
531
+ asChild,
532
+ children,
533
+ isActive = false,
534
+ variant = "default",
535
+ size = "default",
536
+ tooltip,
537
+ className,
538
+ ...props
539
+ }: useRender.ComponentProps<"button"> &
540
+ React.ComponentProps<"button"> & {
541
+ isActive?: boolean
542
+ tooltip?: string | React.ComponentProps<typeof TooltipContent>
543
+ asChild?: boolean
544
+ } & VariantProps<typeof sidebarMenuButtonVariants>) {
545
+ const { isMobile, state } = useSidebar()
546
+ // Convert asChild to render prop for Base UI compatibility
547
+ const renderProp = asChild && React.isValidElement(children) ? children : render
548
+
549
+ const comp = useRender({
550
+ defaultTagName: "button",
551
+ props: mergeProps<"button">(
552
+ {
553
+ className: cn(sidebarMenuButtonVariants({ variant, size }), className),
554
+ children: asChild ? undefined : children,
555
+ },
556
+ props
557
+ ),
558
+ render: !tooltip ? renderProp : TooltipTrigger,
559
+ state: {
560
+ slot: "sidebar-menu-button",
561
+ sidebar: "menu-button",
562
+ size,
563
+ active: isActive,
564
+ },
565
+ })
566
+
567
+ if (!tooltip) {
568
+ return comp
569
+ }
570
+
571
+ if (typeof tooltip === "string") {
572
+ tooltip = {
573
+ children: tooltip,
574
+ }
575
+ }
576
+
577
+ return (
578
+ <Tooltip>
579
+ {comp}
580
+ <TooltipContent
581
+ side="right"
582
+ align="center"
583
+ hidden={state !== "collapsed" || isMobile}
584
+ {...tooltip}
585
+ />
586
+ </Tooltip>
587
+ )
588
+ }
589
+
590
+ function SidebarMenuAction({
591
+ className,
592
+ render,
593
+ asChild,
594
+ children,
595
+ showOnHover = false,
596
+ ...props
597
+ }: useRender.ComponentProps<"button"> &
598
+ React.ComponentProps<"button"> & {
599
+ showOnHover?: boolean
600
+ asChild?: boolean
601
+ }) {
602
+ // Convert asChild to render prop for Base UI compatibility
603
+ const renderProp = asChild && React.isValidElement(children) ? children : render
604
+
605
+ return useRender({
606
+ defaultTagName: "button",
607
+ props: mergeProps<"button">(
608
+ {
609
+ className: cn(
610
+ "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 aspect-square w-5 rounded-md p-0 peer-data-[size=default]/menu-button:top-1.5 peer-data-[size=lg]/menu-button:top-2.5 peer-data-[size=sm]/menu-button:top-1 focus-visible:ring-2 [&>svg]:size-4 flex items-center justify-center outline-hidden transition-transform group-data-[collapsible=icon]:hidden after:absolute after:-inset-2 md:after:hidden [&>svg]:shrink-0",
611
+ showOnHover &&
612
+ "peer-data-active/menu-button:text-sidebar-accent-foreground group-focus-within/menu-item:opacity-100 group-hover/menu-item:opacity-100 data-open:opacity-100 md:opacity-0",
613
+ className
614
+ ),
615
+ children: asChild ? undefined : children,
616
+ },
617
+ props
618
+ ),
619
+ render: renderProp,
620
+ state: {
621
+ slot: "sidebar-menu-action",
622
+ sidebar: "menu-action",
623
+ },
624
+ })
625
+ }
626
+
627
+ function SidebarMenuBadge({
628
+ className,
629
+ ...props
630
+ }: React.ComponentProps<"div">) {
631
+ return (
632
+ <div
633
+ data-slot="sidebar-menu-badge"
634
+ data-sidebar="menu-badge"
635
+ className={cn(
636
+ "text-sidebar-foreground peer-hover/menu-button:text-sidebar-accent-foreground peer-data-active/menu-button:text-sidebar-accent-foreground pointer-events-none absolute right-1 flex h-5 min-w-5 rounded-md px-1 text-xs font-medium peer-data-[size=default]/menu-button:top-1.5 peer-data-[size=lg]/menu-button:top-2.5 peer-data-[size=sm]/menu-button:top-1 flex items-center justify-center tabular-nums select-none group-data-[collapsible=icon]:hidden",
637
+ className
638
+ )}
639
+ {...props}
640
+ />
641
+ )
642
+ }
643
+
644
+ function SidebarMenuSkeleton({
645
+ className,
646
+ showIcon = false,
647
+ ...props
648
+ }: React.ComponentProps<"div"> & {
649
+ showIcon?: boolean
650
+ }) {
651
+ // Random width between 50 to 90%.
652
+ const [width] = React.useState(() => {
653
+ return `${Math.floor(Math.random() * 40) + 50}%`
654
+ })
655
+
656
+ return (
657
+ <div
658
+ data-slot="sidebar-menu-skeleton"
659
+ data-sidebar="menu-skeleton"
660
+ className={cn("h-8 gap-2 rounded-md px-2 flex items-center", className)}
661
+ {...props}
662
+ >
663
+ {showIcon && (
664
+ <Skeleton
665
+ className="size-4 rounded-md"
666
+ data-sidebar="menu-skeleton-icon"
667
+ />
668
+ )}
669
+ <Skeleton
670
+ className="h-4 max-w-(--skeleton-width) flex-1"
671
+ data-sidebar="menu-skeleton-text"
672
+ style={
673
+ {
674
+ "--skeleton-width": width,
675
+ } as React.CSSProperties
676
+ }
677
+ />
678
+ </div>
679
+ )
680
+ }
681
+
682
+ function SidebarMenuSub({ className, ...props }: React.ComponentProps<"ul">) {
683
+ return (
684
+ <ul
685
+ data-slot="sidebar-menu-sub"
686
+ data-sidebar="menu-sub"
687
+ className={cn("border-sidebar-border mx-3.5 translate-x-px gap-1 border-l px-2.5 py-0.5 group-data-[collapsible=icon]:hidden flex min-w-0 flex-col", className)}
688
+ {...props}
689
+ />
690
+ )
691
+ }
692
+
693
+ function SidebarMenuSubItem({
694
+ className,
695
+ ...props
696
+ }: React.ComponentProps<"li">) {
697
+ return (
698
+ <li
699
+ data-slot="sidebar-menu-sub-item"
700
+ data-sidebar="menu-sub-item"
701
+ className={cn("group/menu-sub-item relative", className)}
702
+ {...props}
703
+ />
704
+ )
705
+ }
706
+
707
+ function SidebarMenuSubButton({
708
+ render,
709
+ asChild,
710
+ children,
711
+ size = "md",
712
+ isActive = false,
713
+ className,
714
+ ...props
715
+ }: useRender.ComponentProps<"a"> &
716
+ React.ComponentProps<"a"> & {
717
+ size?: "sm" | "md"
718
+ isActive?: boolean
719
+ asChild?: boolean
720
+ }) {
721
+ // Convert asChild to render prop for Base UI compatibility
722
+ const renderProp = asChild && React.isValidElement(children) ? children : render
723
+
724
+ return useRender({
725
+ defaultTagName: "a",
726
+ props: mergeProps<"a">(
727
+ {
728
+ className: cn(
729
+ "text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground active:bg-sidebar-accent active:text-sidebar-accent-foreground data-active:bg-primary data-active:text-primary-foreground h-7 gap-2 rounded-md px-2 focus-visible:ring-2 data-[size=md]:text-sm data-[size=sm]:text-xs [&>svg]:size-4 flex min-w-0 -translate-x-px items-center overflow-hidden outline-hidden group-data-[collapsible=icon]:hidden disabled:pointer-events-none disabled:opacity-50 aria-disabled:pointer-events-none aria-disabled:opacity-50 [&>span:last-child]:truncate [&>svg]:shrink-0",
730
+ className
731
+ ),
732
+ children: asChild ? undefined : children,
733
+ },
734
+ props
735
+ ),
736
+ render: renderProp,
737
+ state: {
738
+ slot: "sidebar-menu-sub-button",
739
+ sidebar: "menu-sub-button",
740
+ size,
741
+ active: isActive,
742
+ },
743
+ })
744
+ }
745
+
746
+ export {
747
+ Sidebar,
748
+ SidebarContent,
749
+ SidebarFooter,
750
+ SidebarGroup,
751
+ SidebarGroupAction,
752
+ SidebarGroupContent,
753
+ SidebarGroupLabel,
754
+ SidebarHeader,
755
+ SidebarInput,
756
+ SidebarInset,
757
+ SidebarMenu,
758
+ SidebarMenuAction,
759
+ SidebarMenuBadge,
760
+ SidebarMenuButton,
761
+ SidebarMenuItem,
762
+ SidebarMenuSkeleton,
763
+ SidebarMenuSub,
764
+ SidebarMenuSubButton,
765
+ SidebarMenuSubItem,
766
+ SidebarProvider,
767
+ SidebarRail,
768
+ SidebarSeparator,
769
+ SidebarTrigger,
770
+ useSidebar,
771
+ }