includio-cms 0.25.0 → 0.27.0

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 (411) hide show
  1. package/API.md +97 -4
  2. package/CHANGELOG.md +118 -0
  3. package/DOCS.md +1 -1
  4. package/README.md +2 -0
  5. package/ROADMAP.md +14 -0
  6. package/dist/admin/auth-client.d.ts +42 -42
  7. package/dist/admin/client/account/lang.d.ts +1 -0
  8. package/dist/admin/client/account/lang.js +4 -2
  9. package/dist/admin/client/account/profile-section.svelte +2 -2
  10. package/dist/admin/client/account/security-section.svelte +27 -4
  11. package/dist/admin/client/account/sessions-section.svelte +1 -1
  12. package/dist/admin/client/admin/admin-after-login-layout-content.svelte +1 -1
  13. package/dist/admin/client/admin/admin-layout.svelte +12 -2
  14. package/dist/admin/client/admin/admin-layout.svelte.d.ts +2 -1
  15. package/dist/admin/client/admin/dashboard-page.svelte +34 -10
  16. package/dist/admin/client/collection/bulk-actions-bar.svelte +86 -44
  17. package/dist/admin/client/collection/bulk-actions-bar.svelte.d.ts +3 -1
  18. package/dist/admin/client/collection/collection-entries.svelte +52 -36
  19. package/dist/admin/client/collection/collection-entries.svelte.d.ts +3 -0
  20. package/dist/admin/client/collection/collection.svelte +28 -14
  21. package/dist/admin/client/collection/collection.svelte.d.ts +3 -0
  22. package/dist/admin/client/collection/data-table.svelte +240 -130
  23. package/dist/admin/client/collection/data-table.svelte.d.ts +9 -0
  24. package/dist/admin/client/collection/date-cell.svelte +4 -4
  25. package/dist/admin/client/collection/row-actions.svelte +2 -1
  26. package/dist/admin/client/collection/sortable-header.svelte +33 -9
  27. package/dist/admin/client/collection/state-display.svelte +102 -0
  28. package/dist/admin/client/collection/state-display.svelte.d.ts +12 -0
  29. package/dist/admin/client/collection/status-badge.svelte +99 -11
  30. package/dist/admin/client/collection/status-badge.svelte.d.ts +15 -1
  31. package/dist/admin/client/collection/table-pagination.svelte +21 -6
  32. package/dist/admin/client/collection/table-toolbar.svelte +105 -80
  33. package/dist/admin/client/collection/table-toolbar.svelte.d.ts +11 -8
  34. package/dist/admin/client/entry/entry-form.svelte +36 -11
  35. package/dist/admin/client/entry/entry-form.svelte.d.ts +1 -0
  36. package/dist/admin/client/entry/entry-header.svelte +22 -15
  37. package/dist/admin/client/entry/entry-header.svelte.d.ts +1 -0
  38. package/dist/admin/client/entry/entry.svelte +269 -165
  39. package/dist/admin/client/entry/header/a11y-header-badge.svelte +47 -0
  40. package/dist/admin/client/entry/header/a11y-header-badge.svelte.d.ts +8 -0
  41. package/dist/admin/client/entry/header/publish-panel.svelte +69 -13
  42. package/dist/admin/client/entry/header/save-indicator.svelte +57 -28
  43. package/dist/admin/client/entry/header/save-indicator.svelte.d.ts +1 -0
  44. package/dist/admin/client/entry/header/status-badge.svelte +60 -15
  45. package/dist/admin/client/entry/header/status-badge.svelte.d.ts +1 -2
  46. package/dist/admin/client/entry/header/version-history-sheet.svelte +1 -1
  47. package/dist/admin/client/entry/hybrid/hybrid-layout.svelte +74 -23
  48. package/dist/admin/client/entry/hybrid/hybrid-preview.svelte +1 -1
  49. package/dist/admin/client/entry/utils.d.ts +14 -0
  50. package/dist/admin/client/entry/utils.js +28 -0
  51. package/dist/admin/client/form/form-submission/form-submission.svelte +2 -2
  52. package/dist/admin/client/form/form-submissions.svelte +143 -194
  53. package/dist/admin/client/form/form-submissions.svelte.d.ts +2 -0
  54. package/dist/admin/client/login/lang.d.ts +3 -0
  55. package/dist/admin/client/login/lang.js +10 -4
  56. package/dist/admin/client/login/login-form.svelte +8 -1
  57. package/dist/admin/client/login/reset-password-page.svelte +24 -3
  58. package/dist/admin/client/login/schema.d.ts +14 -2
  59. package/dist/admin/client/login/schema.js +19 -8
  60. package/dist/admin/client/maintenance/maintenance-page.svelte +16 -17
  61. package/dist/admin/client/media/media-page.svelte +1 -1
  62. package/dist/admin/client/shop/coupon-edit-page.svelte +117 -13
  63. package/dist/admin/client/shop/coupon-form.svelte +282 -138
  64. package/dist/admin/client/shop/coupon-form.svelte.d.ts +1 -9
  65. package/dist/admin/client/shop/coupon-new-page.svelte +40 -10
  66. package/dist/admin/client/shop/coupon-new-page.svelte.d.ts +2 -17
  67. package/dist/admin/client/shop/coupon-schema.d.ts +28 -0
  68. package/dist/admin/client/shop/coupon-schema.js +53 -0
  69. package/dist/admin/client/shop/coupons-list-page.svelte +262 -118
  70. package/dist/admin/client/shop/coupons-list-page.svelte.d.ts +16 -1
  71. package/dist/admin/client/shop/refund-dialog.svelte +37 -1
  72. package/dist/admin/client/shop/refund-dialog.svelte.d.ts +3 -0
  73. package/dist/admin/client/shop/shipping-method-edit-page.svelte +108 -59
  74. package/dist/admin/client/shop/shipping-method-form.svelte +36 -9
  75. package/dist/admin/client/shop/shipping-method-new-page.svelte +44 -13
  76. package/dist/admin/client/shop/shipping-methods-list-page.svelte +101 -59
  77. package/dist/admin/client/shop/shop-order-detail-page.svelte +220 -84
  78. package/dist/admin/client/shop/shop-orders-list-page.svelte +302 -152
  79. package/dist/admin/client/shop/shop-orders-list-page.svelte.d.ts +18 -1
  80. package/dist/admin/client/shop/shop-products-list-page.svelte +355 -118
  81. package/dist/admin/client/shop/shop-products-list-page.svelte.d.ts +19 -1
  82. package/dist/admin/client/users/accept-invite-page.svelte +24 -3
  83. package/dist/admin/client/users/create-user-dialog.svelte +3 -8
  84. package/dist/admin/client/users/lang.d.ts +2 -0
  85. package/dist/admin/client/users/lang.js +4 -0
  86. package/dist/admin/client/users/pending-invitations.svelte +2 -9
  87. package/dist/admin/client/users/user-name-cell.svelte +20 -0
  88. package/dist/admin/client/users/user-name-cell.svelte.d.ts +9 -0
  89. package/dist/admin/client/users/user-role-badge.svelte +16 -0
  90. package/dist/admin/client/users/user-role-badge.svelte.d.ts +7 -0
  91. package/dist/admin/client/users/user-row-actions.svelte +72 -0
  92. package/dist/admin/client/users/user-row-actions.svelte.d.ts +20 -0
  93. package/dist/admin/client/users/user-sessions-sheet.svelte +2 -11
  94. package/dist/admin/client/users/users-page.svelte +283 -497
  95. package/dist/admin/client/users/users-page.svelte.d.ts +12 -1
  96. package/dist/admin/components/dashboard/form-submissions-widget.svelte +59 -74
  97. package/dist/admin/components/dashboard/recent-activity.svelte +17 -5
  98. package/dist/admin/components/dashboard/recent-entries.svelte +19 -7
  99. package/dist/admin/components/dialogs/confirmation-dialog.svelte +105 -0
  100. package/dist/admin/components/dialogs/confirmation-dialog.svelte.d.ts +13 -0
  101. package/dist/admin/components/fields/block-picker-modal.svelte +6 -0
  102. package/dist/admin/components/fields/blocks-field.svelte +46 -1
  103. package/dist/admin/components/fields/boolean-field.svelte +1 -1
  104. package/dist/admin/components/fields/field-renderer.svelte +29 -22
  105. package/dist/admin/components/fields/file-field.svelte +344 -30
  106. package/dist/admin/components/fields/icon-field.svelte +86 -0
  107. package/dist/admin/components/fields/icon-field.svelte.d.ts +8 -0
  108. package/dist/admin/components/fields/icon-picker-dialog.svelte +174 -0
  109. package/dist/admin/components/fields/icon-picker-dialog.svelte.d.ts +11 -0
  110. package/dist/admin/components/fields/media-field.svelte +16 -2
  111. package/dist/admin/components/fields/object-field.svelte +27 -7
  112. package/dist/admin/components/fields/radio-field.svelte +22 -0
  113. package/dist/admin/components/fields/relation-field.svelte +123 -97
  114. package/dist/admin/components/fields/relation-picker-dialog.svelte +2 -2
  115. package/dist/admin/components/fields/seo-field.svelte +60 -30
  116. package/dist/admin/components/fields/shop-field.svelte +219 -24
  117. package/dist/admin/components/fields/simple-array-field.svelte +321 -151
  118. package/dist/admin/components/fields/simple-array-field.svelte.d.ts +3 -0
  119. package/dist/admin/components/fields/slug-field.svelte +146 -21
  120. package/dist/admin/components/fields/text-field-wrapper.svelte +37 -20
  121. package/dist/admin/components/fields/text-field.svelte +7 -2
  122. package/dist/admin/components/fields/url-field-wrapper.svelte +10 -0
  123. package/dist/admin/components/fields/url-field.svelte +36 -23
  124. package/dist/admin/components/forms/form-error-summary.svelte +143 -0
  125. package/dist/admin/components/forms/form-error-summary.svelte.d.ts +27 -0
  126. package/dist/admin/components/layout/app-sidebar.svelte +7 -2
  127. package/dist/admin/components/layout/detail-page-shell.svelte +71 -0
  128. package/dist/admin/components/layout/detail-page-shell.svelte.d.ts +24 -0
  129. package/dist/admin/components/layout/lang.d.ts +5 -0
  130. package/dist/admin/components/layout/lang.js +10 -0
  131. package/dist/admin/components/layout/layout-renderer.svelte +71 -2
  132. package/dist/admin/components/layout/layout-renderer.svelte.d.ts +1 -0
  133. package/dist/admin/components/layout/layout-tabs.svelte +173 -0
  134. package/dist/admin/components/layout/layout-tabs.svelte.d.ts +24 -0
  135. package/dist/admin/components/layout/nav-breadcrumbs.svelte +25 -7
  136. package/dist/admin/components/layout/nav-collections.svelte +23 -36
  137. package/dist/admin/components/layout/nav-forms.svelte +19 -35
  138. package/dist/admin/components/layout/nav-main.svelte +3 -28
  139. package/dist/admin/components/layout/nav-search.svelte +70 -2
  140. package/dist/admin/components/layout/nav-section.svelte +77 -0
  141. package/dist/admin/components/layout/nav-section.svelte.d.ts +22 -0
  142. package/dist/admin/components/layout/nav-shop.svelte +3 -27
  143. package/dist/admin/components/layout/nav-singletons.svelte +16 -28
  144. package/dist/admin/components/layout/page-header.stories.svelte +93 -0
  145. package/dist/admin/components/layout/page-header.stories.svelte.d.ts +27 -0
  146. package/dist/admin/components/layout/page-header.svelte +68 -0
  147. package/dist/admin/components/layout/page-header.svelte.d.ts +17 -0
  148. package/dist/admin/components/layout/site-header.svelte +9 -0
  149. package/dist/admin/components/layout/site-header.svelte.d.ts +2 -17
  150. package/dist/admin/components/media/file/file-name-input.svelte +6 -2
  151. package/dist/admin/components/media/file/file-preview.svelte +130 -17
  152. package/dist/admin/components/media/file-upload.svelte +16 -7
  153. package/dist/admin/components/media/file-upload.svelte.d.ts +1 -0
  154. package/dist/admin/components/media/files-list.svelte +153 -53
  155. package/dist/admin/components/media/files-list.svelte.d.ts +1 -0
  156. package/dist/admin/components/media/media-library.svelte +577 -198
  157. package/dist/admin/components/media/media-library.svelte.d.ts +4 -0
  158. package/dist/admin/components/media/media-selector.svelte +4 -2
  159. package/dist/admin/components/media/media-selector.svelte.d.ts +1 -0
  160. package/dist/admin/components/media/tag-sidebar.svelte +4 -4
  161. package/dist/admin/components/tiptap/FigureNodeView.svelte +10 -0
  162. package/dist/admin/components/tiptap/bubble-menu.svelte +104 -0
  163. package/dist/admin/components/tiptap/bubble-menu.svelte.d.ts +19 -0
  164. package/dist/admin/components/tiptap/content-editor.svelte +28 -24
  165. package/dist/admin/components/tiptap/editor-toolbar.svelte +7 -7
  166. package/dist/admin/components/tiptap/extensions.js +5 -1
  167. package/dist/admin/components/tiptap/image-dialog.svelte +5 -1
  168. package/dist/admin/components/tiptap/link-dialog.svelte +2 -0
  169. package/dist/admin/components/tiptap/tiptap-editor.svelte +18 -20
  170. package/dist/admin/components/tiptap/video-dialog.svelte +1 -1
  171. package/dist/admin/components/variant-form/VariantAttributeRenderer.svelte +109 -0
  172. package/dist/admin/components/variant-form/VariantAttributeRenderer.svelte.d.ts +9 -0
  173. package/dist/admin/helpers/build-icon-set-map.d.ts +8 -0
  174. package/dist/admin/helpers/build-icon-set-map.js +16 -0
  175. package/dist/admin/helpers/index.d.ts +2 -0
  176. package/dist/admin/helpers/index.js +2 -0
  177. package/dist/admin/i18n/errors.d.ts +140 -0
  178. package/dist/admin/i18n/errors.js +151 -0
  179. package/dist/admin/remote/entry.remote.d.ts +59 -4
  180. package/dist/admin/remote/entry.remote.js +239 -62
  181. package/dist/admin/remote/shop.remote.d.ts +87 -48
  182. package/dist/admin/remote/shop.remote.js +70 -8
  183. package/dist/admin/shared/password-generate.d.ts +6 -0
  184. package/dist/admin/shared/password-generate.js +40 -0
  185. package/dist/admin/shared/password-schema.d.ts +6 -0
  186. package/dist/admin/shared/password-schema.js +10 -3
  187. package/dist/admin/state/icon-sets.svelte.d.ts +9 -0
  188. package/dist/admin/state/icon-sets.svelte.js +20 -0
  189. package/dist/admin/styles/admin.css +23 -6
  190. package/dist/admin/styles/tokens.md +244 -0
  191. package/dist/admin/utils/accordionActivation.d.ts +13 -0
  192. package/dist/admin/utils/accordionActivation.js +35 -0
  193. package/dist/admin/utils/entryLabel.d.ts +23 -0
  194. package/dist/admin/utils/entryLabel.js +51 -12
  195. package/dist/admin/utils/field-a11y.d.ts +29 -0
  196. package/dist/admin/utils/field-a11y.js +23 -0
  197. package/dist/admin/utils/fieldPathElement.d.ts +9 -0
  198. package/dist/admin/utils/fieldPathElement.js +18 -0
  199. package/dist/admin/utils/fileDisplay.d.ts +10 -0
  200. package/dist/admin/utils/fileDisplay.js +26 -0
  201. package/dist/admin/utils/flattenFormErrors.d.ts +19 -0
  202. package/dist/admin/utils/flattenFormErrors.js +102 -0
  203. package/dist/admin/utils/formatters.d.ts +12 -0
  204. package/dist/admin/utils/{formatDate.js → formatters.js} +23 -2
  205. package/dist/admin/utils/scrollWithin.d.ts +9 -0
  206. package/dist/admin/utils/scrollWithin.js +32 -0
  207. package/dist/admin/utils/tabActivation.d.ts +12 -0
  208. package/dist/admin/utils/tabActivation.js +24 -0
  209. package/dist/cli/scaffold/admin.js +2 -2
  210. package/dist/cms/runtime/schema.d.ts +1 -0
  211. package/dist/cms/runtime/schema.js +1 -0
  212. package/dist/cms/runtime/types.d.ts +80 -7
  213. package/dist/components/ui/accordion/accordion-content.svelte +17 -3
  214. package/dist/components/ui/accordion/accordion.stories.svelte +21 -1
  215. package/dist/components/ui/alert/alert.stories.svelte +14 -0
  216. package/dist/components/ui/alert-dialog/alert-dialog.stories.svelte +45 -0
  217. package/dist/components/ui/alert-dialog/alert-dialog.stories.svelte.d.ts +27 -0
  218. package/dist/components/ui/avatar/avatar.stories.svelte +27 -0
  219. package/dist/components/ui/badge/badge.stories.svelte +15 -0
  220. package/dist/components/ui/breadcrumb/breadcrumb.stories.svelte +47 -0
  221. package/dist/components/ui/breadcrumb/breadcrumb.svelte +1 -1
  222. package/dist/components/ui/button/button.stories.svelte +53 -6
  223. package/dist/components/ui/button/button.svelte +39 -5
  224. package/dist/components/ui/button/button.svelte.d.ts +4 -0
  225. package/dist/components/ui/button-group/button-group.stories.svelte +44 -0
  226. package/dist/components/ui/button-group/button-group.stories.svelte.d.ts +27 -0
  227. package/dist/components/ui/calendar/calendar.stories.svelte +36 -0
  228. package/dist/components/ui/calendar/calendar.stories.svelte.d.ts +27 -0
  229. package/dist/components/ui/card/card.stories.svelte +7 -0
  230. package/dist/components/ui/carousel/carousel.stories.svelte +43 -0
  231. package/dist/components/ui/carousel/carousel.stories.svelte.d.ts +27 -0
  232. package/dist/components/ui/checkbox/checkbox.stories.svelte +67 -0
  233. package/dist/components/ui/checkbox/checkbox.stories.svelte.d.ts +27 -0
  234. package/dist/components/ui/checkbox/checkbox.svelte +1 -1
  235. package/dist/components/ui/command/command.stories.svelte +18 -0
  236. package/dist/components/ui/data-table/data-table.stories.svelte +61 -0
  237. package/dist/components/ui/data-table/data-table.stories.svelte.d.ts +18 -0
  238. package/dist/components/ui/dialog/dialog-content.svelte +5 -0
  239. package/dist/components/ui/dialog/dialog-content.svelte.d.ts +2 -0
  240. package/dist/components/ui/dialog/dialog.stories.svelte +35 -0
  241. package/dist/components/ui/dropdown-menu/dropdown-menu.stories.svelte +74 -0
  242. package/dist/components/ui/dropdown-menu/dropdown-menu.stories.svelte.d.ts +27 -0
  243. package/dist/components/ui/field/field-context.svelte.d.ts +22 -0
  244. package/dist/components/ui/field/field-context.svelte.js +9 -0
  245. package/dist/components/ui/field/field-control.svelte +18 -0
  246. package/dist/components/ui/field/field-control.svelte.d.ts +8 -0
  247. package/dist/components/ui/field/field-description.svelte +12 -0
  248. package/dist/components/ui/field/field-error.svelte +14 -6
  249. package/dist/components/ui/field/field-label.svelte +10 -0
  250. package/dist/components/ui/field/field.stories.svelte +95 -9
  251. package/dist/components/ui/field/field.svelte +57 -0
  252. package/dist/components/ui/field/field.svelte.d.ts +2 -0
  253. package/dist/components/ui/field/index.d.ts +3 -1
  254. package/dist/components/ui/field/index.js +4 -2
  255. package/dist/components/ui/form/form-field-errors.svelte +1 -1
  256. package/dist/components/ui/form/form.stories.svelte +25 -0
  257. package/dist/components/ui/form/form.stories.svelte.d.ts +26 -0
  258. package/dist/components/ui/input/input.stories.svelte +26 -0
  259. package/dist/components/ui/input-group/input-group-input.svelte.d.ts +1 -1
  260. package/dist/components/ui/input-group/input-group.stories.svelte +43 -0
  261. package/dist/components/ui/input-group/input-group.stories.svelte.d.ts +27 -0
  262. package/dist/components/ui/item/item.stories.svelte +61 -0
  263. package/dist/components/ui/item/item.stories.svelte.d.ts +27 -0
  264. package/dist/components/ui/label/label.stories.svelte +7 -0
  265. package/dist/components/ui/live-region/index.d.ts +1 -0
  266. package/dist/components/ui/live-region/index.js +1 -0
  267. package/dist/components/ui/live-region/live-region-demo.svelte +32 -0
  268. package/dist/components/ui/live-region/live-region-demo.svelte.d.ts +7 -0
  269. package/dist/components/ui/live-region/live-region.stories.svelte +23 -0
  270. package/dist/components/ui/live-region/live-region.stories.svelte.d.ts +26 -0
  271. package/dist/components/ui/live-region/live-region.svelte +12 -0
  272. package/dist/components/ui/live-region/live-region.svelte.d.ts +8 -0
  273. package/dist/components/ui/popover/popover.stories.svelte +34 -0
  274. package/dist/components/ui/radio-group/radio-group.stories.svelte +58 -0
  275. package/dist/components/ui/radio-group/radio-group.stories.svelte.d.ts +27 -0
  276. package/dist/components/ui/resizable/resizable.stories.svelte +56 -0
  277. package/dist/components/ui/resizable/resizable.stories.svelte.d.ts +27 -0
  278. package/dist/components/ui/select/select.stories.svelte +49 -0
  279. package/dist/components/ui/separator/separator.stories.svelte +18 -0
  280. package/dist/components/ui/sheet/sheet.stories.svelte +34 -0
  281. package/dist/components/ui/sidebar/sidebar-input.svelte.d.ts +1 -1
  282. package/dist/components/ui/sidebar/sidebar-menu-button.svelte +1 -0
  283. package/dist/components/ui/sidebar/sidebar-trigger.svelte +1 -1
  284. package/dist/components/ui/sidebar/sidebar.stories.svelte +72 -0
  285. package/dist/components/ui/sidebar/sidebar.stories.svelte.d.ts +27 -0
  286. package/dist/components/ui/skeleton/skeleton.stories.svelte +39 -0
  287. package/dist/components/ui/skeleton/skeleton.stories.svelte.d.ts +27 -0
  288. package/dist/components/ui/skeleton/skeleton.svelte +6 -0
  289. package/dist/components/ui/sonner/index.d.ts +1 -1
  290. package/dist/components/ui/sonner/index.js +1 -1
  291. package/dist/components/ui/sonner/sonner.stories.svelte +7 -0
  292. package/dist/components/ui/sonner/sonner.svelte +17 -1
  293. package/dist/components/ui/sonner/sonner.svelte.d.ts +6 -0
  294. package/dist/components/ui/spinner/spinner.stories.svelte +30 -0
  295. package/dist/components/ui/spinner/spinner.stories.svelte.d.ts +27 -0
  296. package/dist/components/ui/switch/switch.stories.svelte +56 -0
  297. package/dist/components/ui/switch/switch.stories.svelte.d.ts +27 -0
  298. package/dist/components/ui/table/table-cell.svelte +1 -1
  299. package/dist/components/ui/table/table-head.svelte +1 -1
  300. package/dist/components/ui/table/table.stories.svelte +68 -0
  301. package/dist/components/ui/table/table.stories.svelte.d.ts +27 -0
  302. package/dist/components/ui/table/table.svelte +1 -1
  303. package/dist/components/ui/tabs/tabs.stories.svelte +48 -0
  304. package/dist/components/ui/tabs/tabs.stories.svelte.d.ts +27 -0
  305. package/dist/components/ui/textarea/textarea.stories.svelte +21 -0
  306. package/dist/components/ui/toggle/toggle.stories.svelte +23 -0
  307. package/dist/components/ui/toggle-group/toggle-group.stories.svelte +43 -0
  308. package/dist/components/ui/tooltip/tooltip.stories.svelte +46 -6
  309. package/dist/core/cms.d.ts +11 -2
  310. package/dist/core/cms.js +29 -0
  311. package/dist/core/fields/fieldSchemaToTs.d.ts +7 -0
  312. package/dist/core/fields/fieldSchemaToTs.js +241 -90
  313. package/dist/core/fields/layoutUtils.d.ts +4 -1
  314. package/dist/core/fields/layoutUtils.js +41 -4
  315. package/dist/core/fields/resolveSeo.d.ts +70 -0
  316. package/dist/core/fields/resolveSeo.js +88 -0
  317. package/dist/core/fields/seoFieldDescriptor.d.ts +43 -0
  318. package/dist/core/fields/seoFieldDescriptor.js +74 -0
  319. package/dist/core/fields/slugPath.d.ts +13 -0
  320. package/dist/core/fields/slugPath.js +32 -0
  321. package/dist/core/fields/urlUtils.d.ts +8 -0
  322. package/dist/core/fields/urlUtils.js +27 -0
  323. package/dist/core/index.d.ts +1 -0
  324. package/dist/core/index.js +1 -0
  325. package/dist/core/server/entries/operations/create.js +13 -0
  326. package/dist/core/server/entries/operations/get.d.ts +7 -0
  327. package/dist/core/server/entries/operations/get.js +10 -6
  328. package/dist/core/server/entries/operations/slugUniqueness.d.ts +37 -0
  329. package/dist/core/server/entries/operations/slugUniqueness.js +116 -0
  330. package/dist/core/server/entries/operations/update.d.ts +6 -1
  331. package/dist/core/server/entries/operations/update.js +24 -1
  332. package/dist/core/server/fields/slugResolver.d.ts +3 -13
  333. package/dist/core/server/fields/slugResolver.js +8 -37
  334. package/dist/core/server/generator/fields.d.ts +2 -0
  335. package/dist/core/server/generator/fields.js +44 -18
  336. package/dist/core/server/generator/formFields.js +2 -1
  337. package/dist/core/server/generator/generator.js +6 -5
  338. package/dist/core/server/generator/utils.d.ts +1 -0
  339. package/dist/core/server/generator/utils.js +4 -0
  340. package/dist/db-postgres/schema/shop/order.d.ts +37 -1
  341. package/dist/db-postgres/schema/shop/order.js +3 -1
  342. package/dist/db-postgres/schema/shop/payment.d.ts +20 -0
  343. package/dist/db-postgres/schema/shop/payment.js +4 -1
  344. package/dist/db-postgres/schema/shop/product.d.ts +20 -0
  345. package/dist/db-postgres/schema/shop/product.js +3 -1
  346. package/dist/db-postgres/schema/shop/productVariant.d.ts +12 -2
  347. package/dist/db-postgres/schema/shop/productVariant.js +22 -0
  348. package/dist/shop/cart/types.d.ts +1 -0
  349. package/dist/shop/client/index.d.ts +54 -0
  350. package/dist/shop/client/index.js +5 -1
  351. package/dist/shop/expiry.d.ts +35 -0
  352. package/dist/shop/expiry.js +68 -0
  353. package/dist/shop/http/balance-handler.d.ts +20 -0
  354. package/dist/shop/http/balance-handler.js +91 -0
  355. package/dist/shop/http/cart-handler.js +19 -0
  356. package/dist/shop/http/checkout-handler.js +19 -1
  357. package/dist/shop/http/index.d.ts +2 -0
  358. package/dist/shop/http/index.js +2 -0
  359. package/dist/shop/http/upcoming-handler.d.ts +16 -0
  360. package/dist/shop/http/upcoming-handler.js +65 -0
  361. package/dist/shop/http/webhook-handler.js +46 -9
  362. package/dist/shop/index.d.ts +4 -1
  363. package/dist/shop/index.js +7 -1
  364. package/dist/shop/server/balance-payment.d.ts +40 -0
  365. package/dist/shop/server/balance-payment.js +140 -0
  366. package/dist/shop/server/cart-hydrate.js +2 -0
  367. package/dist/shop/server/init.d.ts +14 -0
  368. package/dist/shop/server/init.js +35 -0
  369. package/dist/shop/server/orders.d.ts +35 -0
  370. package/dist/shop/server/orders.js +155 -2
  371. package/dist/shop/server/payment-policy.d.ts +35 -0
  372. package/dist/shop/server/payment-policy.js +55 -0
  373. package/dist/shop/server/payments.d.ts +29 -0
  374. package/dist/shop/server/payments.js +64 -0
  375. package/dist/shop/server/populate.d.ts +1 -1
  376. package/dist/shop/server/refund.d.ts +17 -12
  377. package/dist/shop/server/refund.js +96 -13
  378. package/dist/shop/server/shop-data.d.ts +6 -1
  379. package/dist/shop/server/shop-data.js +44 -7
  380. package/dist/shop/template.d.ts +13 -0
  381. package/dist/shop/template.js +98 -0
  382. package/dist/shop/types.d.ts +142 -1
  383. package/dist/shop/variant-attributes.d.ts +28 -0
  384. package/dist/shop/variant-attributes.js +69 -0
  385. package/dist/sveltekit/server/handle.js +17 -0
  386. package/dist/sveltekit/server/index.d.ts +1 -0
  387. package/dist/sveltekit/server/index.js +2 -0
  388. package/dist/types/cms.d.ts +4 -3
  389. package/dist/types/cms.schema.d.ts +1 -1
  390. package/dist/types/cms.schema.js +13 -2
  391. package/dist/types/fields.d.ts +56 -2
  392. package/dist/types/index.d.ts +2 -2
  393. package/dist/types/index.js +1 -1
  394. package/dist/types/layout.d.ts +35 -2
  395. package/dist/types/plugins.d.ts +40 -0
  396. package/dist/types/plugins.js +4 -1
  397. package/dist/updates/0.26.0/index.d.ts +2 -0
  398. package/dist/updates/0.26.0/index.js +51 -0
  399. package/dist/updates/0.26.1/index.d.ts +2 -0
  400. package/dist/updates/0.26.1/index.js +19 -0
  401. package/dist/updates/0.27.0/index.d.ts +2 -0
  402. package/dist/updates/0.27.0/index.js +50 -0
  403. package/dist/updates/index.js +7 -1
  404. package/package.json +29 -7
  405. package/dist/admin/client/collection/empty-state.svelte +0 -28
  406. package/dist/admin/client/collection/empty-state.svelte.d.ts +0 -9
  407. package/dist/admin/client/form/submission-status-badge.svelte +0 -41
  408. package/dist/admin/client/form/submission-status-badge.svelte.d.ts +0 -7
  409. package/dist/admin/components/media/file-preview.svelte +0 -51
  410. package/dist/admin/components/media/file-preview.svelte.d.ts +0 -6
  411. package/dist/admin/utils/formatDate.d.ts +0 -5
@@ -6,6 +6,7 @@
6
6
  type RowSelectionState,
7
7
  type PaginationState,
8
8
  type Table as TableType,
9
+ type Column,
9
10
  getCoreRowModel,
10
11
  getFilteredRowModel,
11
12
  getSortedRowModel,
@@ -17,6 +18,7 @@
17
18
  import type { InterfaceLanguage } from '../../../types/languages.js';
18
19
  import { droppable, draggable, dndState } from '@thisux/sveltednd';
19
20
  import { flip } from 'svelte/animate';
21
+ import StateDisplay from './state-display.svelte';
20
22
 
21
23
  type DataTableProps<TData, TValue> = {
22
24
  columns: ColumnDef<TData, TValue>[];
@@ -36,9 +38,18 @@
36
38
  manualPagination?: boolean;
37
39
  pageCount?: number;
38
40
  rowCount?: number;
41
+ /** @deprecated since v0.26.0 — use `emptyTitle`/`emptyDescription` instead. */
39
42
  emptyMessage?: string;
43
+ emptyTitle?: string;
44
+ emptyDescription?: string;
45
+ viewState?: 'idle' | 'loading' | 'error';
46
+ errorMessage?: string;
47
+ loadingMessage?: string;
48
+ onRetry?: () => void;
40
49
  orderable?: boolean;
41
50
  onReorder?: (fromIndex: number, toIndex: number) => void;
51
+ /** Extend table to viewport edges on mobile via negative horizontal margin. Default true. */
52
+ mobileEdgeBleed?: boolean;
42
53
  };
43
54
 
44
55
  let {
@@ -60,16 +71,24 @@
60
71
  pageCount: pageCountProp,
61
72
  rowCount: rowCountProp,
62
73
  emptyMessage,
74
+ emptyTitle,
75
+ emptyDescription,
76
+ viewState = 'idle',
77
+ errorMessage,
78
+ loadingMessage,
79
+ onRetry,
63
80
  orderable = false,
64
- onReorder
81
+ onReorder,
82
+ mobileEdgeBleed = true
65
83
  }: DataTableProps<TData, TValue> = $props();
66
84
 
67
85
  let liveRegionMessage = $state('');
68
86
  let dropProcessing = false;
69
87
 
70
- const matchesReducedMotion = typeof window !== 'undefined'
71
- ? window.matchMedia('(prefers-reduced-motion: reduce)').matches
72
- : false;
88
+ const matchesReducedMotion =
89
+ typeof window !== 'undefined'
90
+ ? window.matchMedia('(prefers-reduced-motion: reduce)').matches
91
+ : false;
73
92
 
74
93
  const interfaceLanguage = useInterfaceLanguage();
75
94
 
@@ -78,6 +97,14 @@
78
97
  pl: 'Brak wyników.'
79
98
  };
80
99
 
100
+ function ariaSortFor(column: Column<TData, unknown>): 'none' | 'ascending' | 'descending' | undefined {
101
+ if (!column.getCanSort()) return undefined;
102
+ const dir = column.getIsSorted();
103
+ if (dir === 'asc') return 'ascending';
104
+ if (dir === 'desc') return 'descending';
105
+ return 'none';
106
+ }
107
+
81
108
  const table = createSvelteTable({
82
109
  get data() {
83
110
  return data;
@@ -124,139 +151,222 @@
124
151
  $effect(() => {
125
152
  tableRef?.(table);
126
153
  });
154
+
155
+ const resolvedEmptyTitle = $derived(
156
+ emptyTitle ?? emptyMessage ?? defaultEmptyMessages[interfaceLanguage.current]
157
+ );
127
158
  </script>
128
159
 
129
- {#if orderable}
130
- <div aria-live="polite" aria-atomic="true" class="sr-only">{liveRegionMessage}</div>
131
- {/if}
160
+ {#if viewState === 'loading' || viewState === 'error'}
161
+ <StateDisplay
162
+ kind={viewState}
163
+ title={viewState === 'loading' ? loadingMessage : errorMessage}
164
+ onCta={viewState === 'error' ? onRetry : undefined}
165
+ />
166
+ {:else}
167
+ {#if orderable}
168
+ <div aria-live="polite" aria-atomic="true" class="sr-only">{liveRegionMessage}</div>
169
+ {/if}
132
170
 
133
- <Table.Root aria-roledescription={orderable ? 'sortable list' : undefined}>
134
- <Table.Header class="bg-muted sticky top-0 z-10">
135
- {#each table.getHeaderGroups() as headerGroup (headerGroup.id)}
136
- <Table.Row class="border-b-0 hover:bg-transparent">
137
- {#if orderable}
138
- <Table.Head class="text-[11px] font-bold uppercase tracking-[0.04em] text-muted-foreground h-10 w-10"></Table.Head>
139
- {/if}
140
- {#each headerGroup.headers as header (header.id)}
141
- <Table.Head class="text-[11px] font-bold uppercase tracking-[0.04em] text-muted-foreground h-10">
142
- {#if !header.isPlaceholder}
143
- <FlexRender
144
- content={header.column.columnDef.header}
145
- context={header.getContext()}
146
- />
147
- {/if}
148
- </Table.Head>
149
- {/each}
150
- </Table.Row>
151
- {/each}
152
- </Table.Header>
153
- <Table.Body>
154
- {#if orderable}
155
- {#each table.getRowModel().rows as row, i (row.id)}
156
- {@const totalRows = table.getRowModel().rows.length}
157
- <tr
158
- use:droppable={{
159
- container: i.toString(),
160
- callbacks: {
161
- onDrop: (state) => {
162
- if (dropProcessing) return;
163
- dropProcessing = true;
164
- const fromIndex = parseInt(state.sourceContainer ?? '');
165
- const toIndex = parseInt(state.targetContainer ?? '');
166
- if (!isNaN(fromIndex) && !isNaN(toIndex)) {
167
- onReorder?.(fromIndex, toIndex);
171
+ <div
172
+ class="data-table-shell"
173
+ data-mobile-bleed={mobileEdgeBleed ? 'true' : undefined}
174
+ >
175
+ <Table.Root aria-roledescription={orderable ? 'sortable list' : undefined}>
176
+ <Table.Header class="bg-muted sticky top-0 z-10">
177
+ {#each table.getHeaderGroups() as headerGroup (headerGroup.id)}
178
+ <Table.Row class="border-b-0 hover:bg-transparent">
179
+ {#if orderable}
180
+ <Table.Head
181
+ class="text-[11px] font-bold uppercase tracking-[0.04em] text-muted-foreground h-10 w-10"
182
+ ></Table.Head>
183
+ {/if}
184
+ {#each headerGroup.headers as header (header.id)}
185
+ <Table.Head
186
+ class="text-[11px] font-bold uppercase tracking-[0.04em] text-muted-foreground h-10"
187
+ aria-sort={ariaSortFor(header.column)}
188
+ >
189
+ {#if !header.isPlaceholder}
190
+ <FlexRender
191
+ content={header.column.columnDef.header}
192
+ context={header.getContext()}
193
+ />
194
+ {/if}
195
+ </Table.Head>
196
+ {/each}
197
+ </Table.Row>
198
+ {/each}
199
+ </Table.Header>
200
+ <Table.Body>
201
+ {#if orderable}
202
+ {#each table.getRowModel().rows as row, i (row.id)}
203
+ {@const totalRows = table.getRowModel().rows.length}
204
+ <tr
205
+ use:droppable={{
206
+ container: i.toString(),
207
+ callbacks: {
208
+ onDrop: (state) => {
209
+ if (dropProcessing) return;
210
+ dropProcessing = true;
211
+ const fromIndex = parseInt(state.sourceContainer ?? '');
212
+ const toIndex = parseInt(state.targetContainer ?? '');
213
+ if (!isNaN(fromIndex) && !isNaN(toIndex)) {
214
+ onReorder?.(fromIndex, toIndex);
215
+ }
216
+ dndState.isDragging = false;
217
+ dndState.draggedItem = null;
218
+ dndState.sourceContainer = '';
219
+ dndState.targetContainer = null;
220
+ dndState.targetElement = null;
221
+ setTimeout(() => {
222
+ dropProcessing = false;
223
+ }, 50);
168
224
  }
169
- dndState.isDragging = false;
170
- dndState.draggedItem = null;
171
- dndState.sourceContainer = '';
172
- dndState.targetContainer = null;
173
- dndState.targetElement = null;
174
- setTimeout(() => { dropProcessing = false; }, 50);
175
225
  }
176
- }
177
- }}
178
- animate:flip={{ duration: matchesReducedMotion ? 0 : 200 }}
179
- data-slot="table-row"
180
- data-state={row.getIsSelected() ? 'selected' : undefined}
181
- class="hover:bg-muted/50 data-[state=selected]:bg-muted border-b transition-colors hover:bg-lavender-lighter/50 data-[state=selected]:bg-lavender-lighter/30 {i === totalRows - 1 ? 'border-b-0' : ''}"
182
- >
183
- <Table.Cell class="py-3 w-10">
184
- <div class="flex items-center gap-1">
185
- <div
186
- use:draggable={{
187
- container: i.toString(),
188
- dragData: { index: i }
189
- }}
190
- class="text-muted-foreground hover:text-foreground cursor-grab"
191
- role="button"
192
- tabindex="0"
193
- aria-label={`Drag to reorder, row ${i + 1} of ${totalRows}`}
194
- aria-roledescription="draggable"
195
- >
196
- <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><circle cx="9" cy="12" r="1"/><circle cx="9" cy="5" r="1"/><circle cx="9" cy="19" r="1"/><circle cx="15" cy="12" r="1"/><circle cx="15" cy="5" r="1"/><circle cx="15" cy="19" r="1"/></svg>
197
- </div>
198
- <div class="flex flex-col">
199
- {#if i > 0}
200
- <button
201
- type="button"
202
- class="p-0.5 rounded text-muted-foreground hover:text-foreground"
203
- aria-label={`Move up, row ${i + 1} of ${totalRows}`}
204
- onclick={() => {
205
- onReorder?.(i, i - 1);
206
- liveRegionMessage = `Moved to position ${i} of ${totalRows}`;
207
- }}
208
- >
209
- <svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="m18 15-6-6-6 6"/></svg>
210
- </button>
211
- {/if}
212
- {#if i < totalRows - 1}
213
- <button
214
- type="button"
215
- class="p-0.5 rounded text-muted-foreground hover:text-foreground"
216
- aria-label={`Move down, row ${i + 1} of ${totalRows}`}
217
- onclick={() => {
218
- onReorder?.(i, i + 1);
219
- liveRegionMessage = `Moved to position ${i + 2} of ${totalRows}`;
220
- }}
226
+ }}
227
+ animate:flip={{ duration: matchesReducedMotion ? 0 : 200 }}
228
+ data-slot="table-row"
229
+ data-state={row.getIsSelected() ? 'selected' : undefined}
230
+ aria-selected={enableSelection && row.getIsSelected() ? true : undefined}
231
+ class="hover:bg-muted/50 data-[state=selected]:bg-muted border-b transition-colors hover:bg-lavender-lighter/50 data-[state=selected]:bg-lavender-lighter/30 {i ===
232
+ totalRows - 1
233
+ ? 'border-b-0'
234
+ : ''}"
235
+ >
236
+ <Table.Cell class="py-3 w-10">
237
+ <div class="flex items-center gap-1">
238
+ <div
239
+ use:draggable={{
240
+ container: i.toString(),
241
+ dragData: { index: i }
242
+ }}
243
+ class="text-muted-foreground hover:text-foreground cursor-grab"
244
+ role="button"
245
+ tabindex="0"
246
+ aria-label={`Drag to reorder, row ${i + 1} of ${totalRows}`}
247
+ aria-roledescription="draggable"
248
+ >
249
+ <svg
250
+ xmlns="http://www.w3.org/2000/svg"
251
+ width="16"
252
+ height="16"
253
+ viewBox="0 0 24 24"
254
+ fill="none"
255
+ stroke="currentColor"
256
+ stroke-width="2"
257
+ stroke-linecap="round"
258
+ stroke-linejoin="round"
259
+ ><circle cx="9" cy="12" r="1" /><circle cx="9" cy="5" r="1" /><circle
260
+ cx="9"
261
+ cy="19"
262
+ r="1"
263
+ /><circle cx="15" cy="12" r="1" /><circle cx="15" cy="5" r="1" /><circle
264
+ cx="15"
265
+ cy="19"
266
+ r="1"
267
+ /></svg
221
268
  >
222
- <svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="m6 9 6 6 6-6"/></svg>
223
- </button>
224
- {/if}
269
+ </div>
270
+ <div class="flex flex-col">
271
+ {#if i > 0}
272
+ <button
273
+ type="button"
274
+ class="p-0.5 rounded text-muted-foreground hover:text-foreground"
275
+ aria-label={`Move up, row ${i + 1} of ${totalRows}`}
276
+ onclick={() => {
277
+ onReorder?.(i, i - 1);
278
+ liveRegionMessage = `Moved to position ${i} of ${totalRows}`;
279
+ }}
280
+ >
281
+ <svg
282
+ xmlns="http://www.w3.org/2000/svg"
283
+ width="14"
284
+ height="14"
285
+ viewBox="0 0 24 24"
286
+ fill="none"
287
+ stroke="currentColor"
288
+ stroke-width="2"><path d="m18 15-6-6-6 6" /></svg
289
+ >
290
+ </button>
291
+ {/if}
292
+ {#if i < totalRows - 1}
293
+ <button
294
+ type="button"
295
+ class="p-0.5 rounded text-muted-foreground hover:text-foreground"
296
+ aria-label={`Move down, row ${i + 1} of ${totalRows}`}
297
+ onclick={() => {
298
+ onReorder?.(i, i + 1);
299
+ liveRegionMessage = `Moved to position ${i + 2} of ${totalRows}`;
300
+ }}
301
+ >
302
+ <svg
303
+ xmlns="http://www.w3.org/2000/svg"
304
+ width="14"
305
+ height="14"
306
+ viewBox="0 0 24 24"
307
+ fill="none"
308
+ stroke="currentColor"
309
+ stroke-width="2"><path d="m6 9 6 6 6-6" /></svg
310
+ >
311
+ </button>
312
+ {/if}
313
+ </div>
225
314
  </div>
226
- </div>
227
- </Table.Cell>
228
- {#each row.getVisibleCells() as cell (cell.id)}
229
- <Table.Cell class="py-3">
230
- <FlexRender content={cell.column.columnDef.cell} context={cell.getContext()} />
231
315
  </Table.Cell>
232
- {/each}
233
- </tr>
234
- {:else}
235
- <Table.Row>
236
- <Table.Cell colspan={columns.length + 1} class="h-24 text-center text-muted-foreground">
237
- {emptyMessage ?? defaultEmptyMessages[interfaceLanguage.current]}
238
- </Table.Cell>
239
- </Table.Row>
240
- {/each}
241
- {:else}
242
- {#each table.getRowModel().rows as row, i (row.id)}
243
- <Table.Row
244
- data-state={row.getIsSelected() && 'selected'}
245
- class="hover:bg-lavender-lighter/50 data-[state=selected]:bg-lavender-lighter/30 {i === table.getRowModel().rows.length - 1 ? 'border-b-0' : ''}"
246
- >
247
- {#each row.getVisibleCells() as cell (cell.id)}
248
- <Table.Cell class="py-3">
249
- <FlexRender content={cell.column.columnDef.cell} context={cell.getContext()} />
316
+ {#each row.getVisibleCells() as cell (cell.id)}
317
+ <Table.Cell class="py-3">
318
+ <FlexRender content={cell.column.columnDef.cell} context={cell.getContext()} />
319
+ </Table.Cell>
320
+ {/each}
321
+ </tr>
322
+ {:else}
323
+ <Table.Row>
324
+ <Table.Cell
325
+ colspan={columns.length + 1}
326
+ class="h-24 text-center text-muted-foreground p-0"
327
+ >
328
+ <StateDisplay kind="empty" title={resolvedEmptyTitle} description={emptyDescription} />
250
329
  </Table.Cell>
251
- {/each}
252
- </Table.Row>
330
+ </Table.Row>
331
+ {/each}
253
332
  {:else}
254
- <Table.Row>
255
- <Table.Cell colspan={columns.length} class="h-24 text-center text-muted-foreground">
256
- {emptyMessage ?? defaultEmptyMessages[interfaceLanguage.current]}
257
- </Table.Cell>
258
- </Table.Row>
259
- {/each}
260
- {/if}
261
- </Table.Body>
262
- </Table.Root>
333
+ {#each table.getRowModel().rows as row, i (row.id)}
334
+ <Table.Row
335
+ data-state={row.getIsSelected() && 'selected'}
336
+ aria-selected={enableSelection && row.getIsSelected() ? true : undefined}
337
+ class="hover:bg-lavender-lighter/50 data-[state=selected]:bg-lavender-lighter/30 {i ===
338
+ table.getRowModel().rows.length - 1
339
+ ? 'border-b-0'
340
+ : ''}"
341
+ >
342
+ {#each row.getVisibleCells() as cell (cell.id)}
343
+ <Table.Cell class="py-3">
344
+ <FlexRender content={cell.column.columnDef.cell} context={cell.getContext()} />
345
+ </Table.Cell>
346
+ {/each}
347
+ </Table.Row>
348
+ {:else}
349
+ <Table.Row>
350
+ <Table.Cell colspan={columns.length} class="h-24 text-center text-muted-foreground p-0">
351
+ <StateDisplay kind="empty" title={resolvedEmptyTitle} description={emptyDescription} />
352
+ </Table.Cell>
353
+ </Table.Row>
354
+ {/each}
355
+ {/if}
356
+ </Table.Body>
357
+ </Table.Root>
358
+ </div>
359
+ {/if}
360
+
361
+ <style>
362
+ /* Mobile edge bleed: table can scroll edge-to-edge on small screens. */
363
+ @media (max-width: 639px) {
364
+ .data-table-shell[data-mobile-bleed='true'] :global([data-slot='table-container']) {
365
+ margin-left: -1rem;
366
+ margin-right: -1rem;
367
+ border-left: 0;
368
+ border-right: 0;
369
+ border-radius: 0;
370
+ }
371
+ }
372
+ </style>
@@ -17,9 +17,18 @@ type DataTableProps<TData, TValue> = {
17
17
  manualPagination?: boolean;
18
18
  pageCount?: number;
19
19
  rowCount?: number;
20
+ /** @deprecated since v0.26.0 — use `emptyTitle`/`emptyDescription` instead. */
20
21
  emptyMessage?: string;
22
+ emptyTitle?: string;
23
+ emptyDescription?: string;
24
+ viewState?: 'idle' | 'loading' | 'error';
25
+ errorMessage?: string;
26
+ loadingMessage?: string;
27
+ onRetry?: () => void;
21
28
  orderable?: boolean;
22
29
  onReorder?: (fromIndex: number, toIndex: number) => void;
30
+ /** Extend table to viewport edges on mobile via negative horizontal margin. Default true. */
31
+ mobileEdgeBleed?: boolean;
23
32
  };
24
33
  declare function $$render<TData, TValue>(): {
25
34
  props: DataTableProps<TData, TValue>;
@@ -1,5 +1,5 @@
1
1
  <script lang="ts">
2
- import { formatRelativeDate, formatAbsoluteDate, formatShortDate } from '../../utils/formatDate.js';
2
+ import { formatRelativeDate, formatDateTime, formatShortDateOnly } from '../../utils/formatters.js';
3
3
  import { useInterfaceLanguage } from '../../state/interface-language.svelte.js';
4
4
 
5
5
  type Props = {
@@ -15,11 +15,11 @@
15
15
  format === 'relative'
16
16
  ? formatRelativeDate(date, interfaceLanguage.current)
17
17
  : format === 'short'
18
- ? formatShortDate(date, interfaceLanguage.current)
19
- : formatAbsoluteDate(date, interfaceLanguage.current)
18
+ ? formatShortDateOnly(date, interfaceLanguage.current)
19
+ : formatDateTime(date, interfaceLanguage.current)
20
20
  );
21
21
 
22
- const absoluteDate = $derived(formatAbsoluteDate(date, interfaceLanguage.current));
22
+ const absoluteDate = $derived(formatDateTime(date, interfaceLanguage.current));
23
23
  </script>
24
24
 
25
25
  <span title={absoluteDate} class="text-muted-foreground text-sm whitespace-nowrap">
@@ -7,6 +7,7 @@
7
7
  import Button from '../../../components/ui/button/button.svelte';
8
8
  import * as DropdownMenu from '../../../components/ui/dropdown-menu/index.js';
9
9
  import { useInterfaceLanguage } from '../../state/interface-language.svelte.js';
10
+ import { goto } from '$app/navigation';
10
11
  import type { InterfaceLanguage } from '../../../types/languages.js';
11
12
 
12
13
  type Props = {
@@ -45,7 +46,7 @@
45
46
  {/snippet}
46
47
  </DropdownMenu.Trigger>
47
48
  <DropdownMenu.Content align="end" class="w-40">
48
- <DropdownMenu.Item href={entryUrl}>
49
+ <DropdownMenu.Item onclick={() => goto(entryUrl)}>
49
50
  <ExternalLink class="mr-2 size-4" />
50
51
  {t.open}
51
52
  </DropdownMenu.Item>
@@ -4,6 +4,8 @@
4
4
  import ArrowDown from '@tabler/icons-svelte/icons/arrow-down';
5
5
  import ArrowsSort from '@tabler/icons-svelte/icons/arrows-sort';
6
6
  import Button from '../../../components/ui/button/button.svelte';
7
+ import { useInterfaceLanguage } from '../../state/interface-language.svelte.js';
8
+ import type { InterfaceLanguage } from '../../../types/languages.js';
7
9
 
8
10
  type Props = {
9
11
  column: Column<TData, unknown>;
@@ -13,24 +15,45 @@
13
15
 
14
16
  let { column, label, sorting = [] }: Props = $props();
15
17
 
16
- const sortDir = $derived(sorting.find((s) => s.id === column.id)?.desc ? 'desc' : sorting.find((s) => s.id === column.id) ? 'asc' : false);
18
+ const interfaceLanguage = useInterfaceLanguage();
19
+
20
+ const lang: Record<InterfaceLanguage, { asc: string; desc: string; unsorted: string }> = {
21
+ en: {
22
+ asc: 'sorted ascending',
23
+ desc: 'sorted descending',
24
+ unsorted: 'not sorted'
25
+ },
26
+ pl: {
27
+ asc: 'sortowane rosnąco',
28
+ desc: 'sortowane malejąco',
29
+ unsorted: 'nieposortowane'
30
+ }
31
+ };
32
+
33
+ const t = $derived(lang[interfaceLanguage.current]);
34
+
35
+ const sortDir = $derived(
36
+ sorting.find((s) => s.id === column.id)?.desc
37
+ ? 'desc'
38
+ : sorting.find((s) => s.id === column.id)
39
+ ? 'asc'
40
+ : false
41
+ );
42
+
43
+ const srLabel = $derived(sortDir === 'asc' ? t.asc : sortDir === 'desc' ? t.desc : t.unsorted);
17
44
 
18
45
  function handleClick() {
19
46
  if (!sortDir) {
20
- column.toggleSorting(false); // asc
47
+ column.toggleSorting(false);
21
48
  } else if (sortDir === 'asc') {
22
- column.toggleSorting(true); // desc
49
+ column.toggleSorting(true);
23
50
  } else {
24
- column.clearSorting(); // clear
51
+ column.clearSorting();
25
52
  }
26
53
  }
27
54
  </script>
28
55
 
29
- <Button
30
- variant="ghost"
31
- class="-ml-3 h-8 px-2 hover:bg-transparent"
32
- onclick={handleClick}
33
- >
56
+ <Button variant="ghost" class="-ml-3 h-8 px-2 hover:bg-transparent" onclick={handleClick}>
34
57
  {label}
35
58
  {#if sortDir === 'asc'}
36
59
  <ArrowUp class="ml-1 size-4" />
@@ -39,4 +62,5 @@
39
62
  {:else}
40
63
  <ArrowsSort class="ml-1 size-4 opacity-50" />
41
64
  {/if}
65
+ <span class="sr-only"> ({srLabel})</span>
42
66
  </Button>
@@ -0,0 +1,102 @@
1
+ <script lang="ts">
2
+ import type { Snippet } from 'svelte';
3
+ import FileText from '@tabler/icons-svelte/icons/file-text';
4
+ import Loader2 from '@tabler/icons-svelte/icons/loader-2';
5
+ import AlertCircle from '@tabler/icons-svelte/icons/alert-circle';
6
+ import Plus from '@tabler/icons-svelte/icons/plus';
7
+ import Refresh from '@tabler/icons-svelte/icons/refresh';
8
+ import Button from '../../../components/ui/button/button.svelte';
9
+ import { useInterfaceLanguage } from '../../state/interface-language.svelte.js';
10
+ import { errorMessages, pickError } from '../../i18n/errors.js';
11
+ import type { InterfaceLanguage } from '../../../types/languages.js';
12
+
13
+ type Props = {
14
+ kind: 'empty' | 'loading' | 'error';
15
+ title?: string;
16
+ description?: string;
17
+ ctaLabel?: string;
18
+ onCta?: () => void;
19
+ icon?: Snippet;
20
+ };
21
+
22
+ let { kind, title, description, ctaLabel, onCta, icon }: Props = $props();
23
+
24
+ const interfaceLanguage = useInterfaceLanguage();
25
+
26
+ const defaults: Record<InterfaceLanguage, { loading: string; error: string; empty: string; retry: string }> = {
27
+ en: {
28
+ loading: 'Loading…',
29
+ error: pickError(errorMessages.network.generic, 'en'),
30
+ empty: 'Nothing here yet',
31
+ retry: 'Try again'
32
+ },
33
+ pl: {
34
+ loading: 'Ładowanie…',
35
+ error: pickError(errorMessages.network.generic, 'pl'),
36
+ empty: 'Nic tu jeszcze nie ma',
37
+ retry: 'Spróbuj ponownie'
38
+ }
39
+ };
40
+
41
+ const t = $derived(defaults[interfaceLanguage.current]);
42
+ const resolvedTitle = $derived(title ?? (kind === 'loading' ? t.loading : kind === 'error' ? t.error : t.empty));
43
+ const resolvedCtaLabel = $derived(ctaLabel ?? (kind === 'error' ? t.retry : undefined));
44
+ </script>
45
+
46
+ {#if kind === 'loading'}
47
+ <div
48
+ class="flex flex-col items-center justify-center py-16 px-4 text-center"
49
+ role="status"
50
+ aria-live="polite"
51
+ aria-busy="true"
52
+ >
53
+ {#if icon}{@render icon()}{:else}
54
+ <div class="flex items-center justify-center w-[72px] h-[72px] rounded-2xl bg-muted mb-5">
55
+ <Loader2 class="size-8 text-muted-foreground animate-spin" />
56
+ </div>
57
+ {/if}
58
+ <h3 class="text-lg font-bold text-foreground mb-2">{resolvedTitle}</h3>
59
+ {#if description}
60
+ <p class="text-sm text-muted-foreground max-w-[360px]">{description}</p>
61
+ {/if}
62
+ </div>
63
+ {:else if kind === 'error'}
64
+ <div
65
+ class="flex flex-col items-center justify-center py-16 px-4 text-center"
66
+ role="alert"
67
+ >
68
+ {#if icon}{@render icon()}{:else}
69
+ <div class="flex items-center justify-center w-[72px] h-[72px] rounded-2xl bg-destructive-bg mb-5">
70
+ <AlertCircle class="size-8 text-destructive" />
71
+ </div>
72
+ {/if}
73
+ <h3 class="text-lg font-bold text-foreground mb-2">{resolvedTitle}</h3>
74
+ {#if description}
75
+ <p class="text-sm text-muted-foreground max-w-[360px] mb-6">{description}</p>
76
+ {/if}
77
+ {#if resolvedCtaLabel && onCta}
78
+ <Button variant="default" onclick={onCta}>
79
+ <Refresh class="size-4 mr-1.5" />
80
+ {resolvedCtaLabel}
81
+ </Button>
82
+ {/if}
83
+ </div>
84
+ {:else}
85
+ <div class="flex flex-col items-center justify-center py-16 px-4 text-center">
86
+ {#if icon}{@render icon()}{:else}
87
+ <div class="flex items-center justify-center w-[72px] h-[72px] rounded-2xl bg-muted mb-5">
88
+ <FileText class="size-8 text-muted-foreground" />
89
+ </div>
90
+ {/if}
91
+ <h3 class="text-lg font-bold text-foreground mb-2">{resolvedTitle}</h3>
92
+ {#if description}
93
+ <p class="text-sm text-muted-foreground max-w-[360px] mb-6">{description}</p>
94
+ {/if}
95
+ {#if resolvedCtaLabel && onCta}
96
+ <Button variant="default" onclick={onCta} data-testid="create-entry-button">
97
+ <Plus class="size-4 mr-1.5" />
98
+ {resolvedCtaLabel}
99
+ </Button>
100
+ {/if}
101
+ </div>
102
+ {/if}