@spaceinvoices/react-ui 0.4.11 → 0.4.12

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 (633) hide show
  1. package/cli/dist/index.js +1 -1
  2. package/package.json +3 -2
  3. package/registry.json +30 -31
  4. package/src/common/autocomplete.tsx +41 -25
  5. package/src/components/activities/locales/bg.ts +21 -0
  6. package/src/components/activities/locales/cs.ts +21 -0
  7. package/src/components/activities/locales/en.ts +21 -0
  8. package/src/components/activities/locales/et.ts +21 -0
  9. package/src/components/activities/locales/fi.ts +21 -0
  10. package/src/components/activities/locales/is.ts +21 -0
  11. package/src/components/activities/locales/nb.ts +21 -0
  12. package/src/components/activities/locales/sk.ts +21 -0
  13. package/src/components/activities/locales/sv.ts +21 -0
  14. package/src/components/advance-invoices/advance-invoices.hooks.ts +2 -3
  15. package/src/components/advance-invoices/create/create-advance-invoice-form.tsx +290 -69
  16. package/src/components/advance-invoices/create/locales/bg.ts +48 -0
  17. package/src/components/advance-invoices/create/locales/cs.ts +48 -0
  18. package/src/components/advance-invoices/create/locales/en.ts +48 -0
  19. package/src/components/advance-invoices/create/locales/et.ts +48 -0
  20. package/src/components/advance-invoices/create/locales/fi.ts +48 -0
  21. package/src/components/advance-invoices/create/locales/hr.ts +17 -17
  22. package/src/components/advance-invoices/create/locales/is.ts +48 -0
  23. package/src/components/advance-invoices/create/locales/nb.ts +48 -0
  24. package/src/components/advance-invoices/create/locales/sk.ts +48 -0
  25. package/src/components/advance-invoices/create/locales/sl.ts +9 -7
  26. package/src/components/advance-invoices/create/locales/sv.ts +48 -0
  27. package/src/components/advance-invoices/create/prepare-advance-invoice-submission.ts +16 -2
  28. package/src/components/advance-invoices/list/list-row-actions.tsx +48 -16
  29. package/src/components/advance-invoices/list/list-table.tsx +33 -8
  30. package/src/components/advance-invoices/list/locales/bg.ts +51 -0
  31. package/src/components/advance-invoices/list/locales/cs.ts +51 -0
  32. package/src/components/advance-invoices/list/locales/en.ts +33 -2
  33. package/src/components/advance-invoices/list/locales/et.ts +51 -0
  34. package/src/components/advance-invoices/list/locales/fi.ts +51 -0
  35. package/src/components/advance-invoices/list/locales/hr.ts +15 -15
  36. package/src/components/advance-invoices/list/locales/is.ts +51 -0
  37. package/src/components/advance-invoices/list/locales/nb.ts +51 -0
  38. package/src/components/advance-invoices/list/locales/sk.ts +51 -0
  39. package/src/components/advance-invoices/list/locales/sv.ts +51 -0
  40. package/src/components/advance-invoices/list/use-advance-invoice-download.ts +3 -5
  41. package/src/components/common/autocomplete-locales.ts +101 -0
  42. package/src/components/company-registry/company-registry.hooks.ts +3 -7
  43. package/src/components/credit-notes/create/create-credit-note-form.tsx +211 -22
  44. package/src/components/credit-notes/create/locales/bg.ts +87 -0
  45. package/src/components/credit-notes/create/locales/cs.ts +87 -0
  46. package/src/components/credit-notes/create/locales/en.ts +87 -0
  47. package/src/components/credit-notes/create/locales/et.ts +87 -0
  48. package/src/components/credit-notes/create/locales/fi.ts +87 -0
  49. package/src/components/credit-notes/create/locales/hr.ts +24 -24
  50. package/src/components/credit-notes/create/locales/is.ts +87 -0
  51. package/src/components/credit-notes/create/locales/nb.ts +87 -0
  52. package/src/components/credit-notes/create/locales/sk.ts +87 -0
  53. package/src/components/credit-notes/create/locales/sl.ts +9 -9
  54. package/src/components/credit-notes/create/locales/sv.ts +87 -0
  55. package/src/components/credit-notes/create/prepare-credit-note-submission.ts +32 -0
  56. package/src/components/credit-notes/credit-notes.hooks.ts +15 -3
  57. package/src/components/credit-notes/list/list-row-actions.tsx +19 -7
  58. package/src/components/credit-notes/list/list-table.tsx +58 -7
  59. package/src/components/credit-notes/list/locales/bg.ts +48 -0
  60. package/src/components/credit-notes/list/locales/cs.ts +48 -0
  61. package/src/components/credit-notes/list/locales/en.ts +31 -2
  62. package/src/components/credit-notes/list/locales/et.ts +48 -0
  63. package/src/components/credit-notes/list/locales/fi.ts +48 -0
  64. package/src/components/credit-notes/list/locales/hr.ts +12 -12
  65. package/src/components/credit-notes/list/locales/is.ts +48 -0
  66. package/src/components/credit-notes/list/locales/nb.ts +48 -0
  67. package/src/components/credit-notes/list/locales/sk.ts +48 -0
  68. package/src/components/credit-notes/list/locales/sv.ts +48 -0
  69. package/src/components/credit-notes/list/use-credit-note-download.ts +3 -5
  70. package/src/components/customers/create-customer-form/locales/bg.ts +21 -0
  71. package/src/components/customers/create-customer-form/locales/cs.ts +21 -0
  72. package/src/components/customers/create-customer-form/locales/en.ts +21 -0
  73. package/src/components/customers/create-customer-form/locales/et.ts +21 -0
  74. package/src/components/customers/create-customer-form/locales/fi.ts +21 -0
  75. package/src/components/customers/create-customer-form/locales/hr.ts +11 -11
  76. package/src/components/customers/create-customer-form/locales/is.ts +21 -0
  77. package/src/components/customers/create-customer-form/locales/nb.ts +21 -0
  78. package/src/components/customers/create-customer-form/locales/sk.ts +21 -0
  79. package/src/components/customers/create-customer-form/locales/sv.ts +21 -0
  80. package/src/components/customers/customer-autocomplete.tsx +32 -4
  81. package/src/components/customers/customer-list-table/customer-list-table.tsx +8 -6
  82. package/src/components/customers/customer-list-table/locales/bg.ts +25 -0
  83. package/src/components/customers/customer-list-table/locales/cs.ts +25 -0
  84. package/src/components/customers/customer-list-table/locales/en.ts +17 -2
  85. package/src/components/customers/customer-list-table/locales/et.ts +25 -0
  86. package/src/components/customers/customer-list-table/locales/fi.ts +25 -0
  87. package/src/components/customers/customer-list-table/locales/hr.ts +11 -11
  88. package/src/components/customers/customer-list-table/locales/is.ts +25 -0
  89. package/src/components/customers/customer-list-table/locales/nb.ts +25 -0
  90. package/src/components/customers/customer-list-table/locales/sk.ts +25 -0
  91. package/src/components/customers/customer-list-table/locales/sv.ts +25 -0
  92. package/src/components/customers/customers.hooks.ts +14 -17
  93. package/src/components/customers/edit-customer-form/locales/bg.ts +13 -0
  94. package/src/components/customers/edit-customer-form/locales/cs.ts +13 -0
  95. package/src/components/customers/edit-customer-form/locales/en.ts +13 -0
  96. package/src/components/customers/edit-customer-form/locales/et.ts +13 -0
  97. package/src/components/customers/edit-customer-form/locales/fi.ts +13 -0
  98. package/src/components/customers/edit-customer-form/locales/hr.ts +5 -5
  99. package/src/components/customers/edit-customer-form/locales/is.ts +13 -0
  100. package/src/components/customers/edit-customer-form/locales/nb.ts +13 -0
  101. package/src/components/customers/edit-customer-form/locales/sk.ts +13 -0
  102. package/src/components/customers/edit-customer-form/locales/sv.ts +13 -0
  103. package/src/components/dashboard/collection-rate-card/collection-rate-card.tsx +7 -6
  104. package/src/components/dashboard/collection-rate-card/locales/en.ts +3 -0
  105. package/src/components/dashboard/collection-rate-card/use-collection-rate.ts +8 -7
  106. package/src/components/dashboard/invoice-status-chart/invoice-status-chart.tsx +7 -4
  107. package/src/components/dashboard/invoice-status-chart/locales/bg.ts +5 -5
  108. package/src/components/dashboard/invoice-status-chart/locales/cs.ts +5 -5
  109. package/src/components/dashboard/invoice-status-chart/locales/de.ts +1 -1
  110. package/src/components/dashboard/invoice-status-chart/locales/en.ts +10 -0
  111. package/src/components/dashboard/invoice-status-chart/locales/es.ts +1 -1
  112. package/src/components/dashboard/invoice-status-chart/locales/et.ts +5 -5
  113. package/src/components/dashboard/invoice-status-chart/locales/fi.ts +5 -5
  114. package/src/components/dashboard/invoice-status-chart/locales/fr.ts +1 -1
  115. package/src/components/dashboard/invoice-status-chart/locales/hr.ts +5 -5
  116. package/src/components/dashboard/invoice-status-chart/locales/is.ts +5 -5
  117. package/src/components/dashboard/invoice-status-chart/locales/it.ts +1 -1
  118. package/src/components/dashboard/invoice-status-chart/locales/nb.ts +5 -5
  119. package/src/components/dashboard/invoice-status-chart/locales/nl.ts +1 -1
  120. package/src/components/dashboard/invoice-status-chart/locales/pl.ts +1 -1
  121. package/src/components/dashboard/invoice-status-chart/locales/pt.ts +1 -1
  122. package/src/components/dashboard/invoice-status-chart/locales/sk.ts +5 -5
  123. package/src/components/dashboard/invoice-status-chart/locales/sl.ts +1 -1
  124. package/src/components/dashboard/invoice-status-chart/locales/sv.ts +5 -5
  125. package/src/components/dashboard/payment-methods-chart/locales/bg.ts +5 -5
  126. package/src/components/dashboard/payment-methods-chart/locales/cs.ts +5 -5
  127. package/src/components/dashboard/payment-methods-chart/locales/en.ts +12 -0
  128. package/src/components/dashboard/payment-methods-chart/locales/et.ts +5 -5
  129. package/src/components/dashboard/payment-methods-chart/locales/fi.ts +6 -6
  130. package/src/components/dashboard/payment-methods-chart/locales/hr.ts +3 -3
  131. package/src/components/dashboard/payment-methods-chart/locales/is.ts +5 -5
  132. package/src/components/dashboard/payment-methods-chart/locales/nb.ts +5 -5
  133. package/src/components/dashboard/payment-methods-chart/locales/sk.ts +5 -5
  134. package/src/components/dashboard/payment-methods-chart/locales/sv.ts +5 -5
  135. package/src/components/dashboard/payment-methods-chart/payment-methods-chart.tsx +4 -3
  136. package/src/components/dashboard/payment-trend-chart/locales/bg.ts +1 -1
  137. package/src/components/dashboard/payment-trend-chart/locales/cs.ts +1 -1
  138. package/src/components/dashboard/payment-trend-chart/locales/en.ts +6 -0
  139. package/src/components/dashboard/payment-trend-chart/locales/et.ts +1 -1
  140. package/src/components/dashboard/payment-trend-chart/locales/fi.ts +1 -1
  141. package/src/components/dashboard/payment-trend-chart/locales/is.ts +1 -1
  142. package/src/components/dashboard/payment-trend-chart/locales/nb.ts +1 -1
  143. package/src/components/dashboard/payment-trend-chart/locales/sk.ts +1 -1
  144. package/src/components/dashboard/payment-trend-chart/locales/sv.ts +1 -1
  145. package/src/components/dashboard/payment-trend-chart/payment-trend-chart.tsx +4 -3
  146. package/src/components/dashboard/payment-trend-chart/use-payment-trend.ts +12 -9
  147. package/src/components/dashboard/revenue-card.tsx +3 -3
  148. package/src/components/dashboard/revenue-trend-chart/locales/bg.ts +1 -1
  149. package/src/components/dashboard/revenue-trend-chart/locales/cs.ts +1 -1
  150. package/src/components/dashboard/revenue-trend-chart/locales/en.ts +6 -0
  151. package/src/components/dashboard/revenue-trend-chart/locales/et.ts +1 -1
  152. package/src/components/dashboard/revenue-trend-chart/locales/fi.ts +1 -1
  153. package/src/components/dashboard/revenue-trend-chart/locales/is.ts +1 -1
  154. package/src/components/dashboard/revenue-trend-chart/locales/nb.ts +1 -1
  155. package/src/components/dashboard/revenue-trend-chart/locales/sk.ts +1 -1
  156. package/src/components/dashboard/revenue-trend-chart/locales/sv.ts +1 -1
  157. package/src/components/dashboard/revenue-trend-chart/revenue-trend-chart.tsx +4 -3
  158. package/src/components/dashboard/revenue-trend-chart/use-revenue-trend.ts +4 -3
  159. package/src/components/dashboard/shared/local-date.ts +11 -0
  160. package/src/components/dashboard/shared/use-revenue-data.ts +6 -5
  161. package/src/components/dashboard/shared/use-stats-query.ts +7 -11
  162. package/src/components/dashboard/stat-card.tsx +3 -3
  163. package/src/components/dashboard/tax-collected-card/tax-collected-card.tsx +11 -9
  164. package/src/components/dashboard/tax-collected-card/use-tax-collected.ts +3 -2
  165. package/src/components/dashboard/top-customers-chart/locales/bg.ts +2 -2
  166. package/src/components/dashboard/top-customers-chart/locales/cs.ts +2 -2
  167. package/src/components/dashboard/top-customers-chart/locales/en.ts +7 -0
  168. package/src/components/dashboard/top-customers-chart/locales/et.ts +2 -2
  169. package/src/components/dashboard/top-customers-chart/locales/fi.ts +2 -2
  170. package/src/components/dashboard/top-customers-chart/locales/is.ts +2 -2
  171. package/src/components/dashboard/top-customers-chart/locales/nb.ts +2 -2
  172. package/src/components/dashboard/top-customers-chart/locales/sk.ts +2 -2
  173. package/src/components/dashboard/top-customers-chart/locales/sv.ts +2 -2
  174. package/src/components/dashboard/top-customers-chart/top-customers-chart.tsx +4 -3
  175. package/src/components/dashboard/top-customers-chart/use-top-customers.ts +7 -20
  176. package/src/components/delivery-notes/create/create-delivery-note-form.tsx +61 -31
  177. package/src/components/delivery-notes/create/locales/bg.ts +68 -0
  178. package/src/components/delivery-notes/create/locales/cs.ts +68 -0
  179. package/src/components/delivery-notes/create/locales/en.ts +68 -0
  180. package/src/components/delivery-notes/create/locales/et.ts +68 -0
  181. package/src/components/delivery-notes/create/locales/fi.ts +68 -0
  182. package/src/components/delivery-notes/create/locales/hr.ts +6 -6
  183. package/src/components/delivery-notes/create/locales/is.ts +68 -0
  184. package/src/components/delivery-notes/create/locales/nb.ts +68 -0
  185. package/src/components/delivery-notes/create/locales/sk.ts +68 -0
  186. package/src/components/delivery-notes/create/locales/sl.ts +5 -5
  187. package/src/components/delivery-notes/create/locales/sv.ts +68 -0
  188. package/src/components/delivery-notes/delivery-notes.hooks.ts +20 -3
  189. package/src/components/delivery-notes/list/list-row-actions.tsx +62 -16
  190. package/src/components/delivery-notes/list/list-table.tsx +42 -9
  191. package/src/components/delivery-notes/list/locales/bg.ts +40 -0
  192. package/src/components/delivery-notes/list/locales/cs.ts +40 -0
  193. package/src/components/delivery-notes/list/locales/et.ts +40 -0
  194. package/src/components/delivery-notes/list/locales/fi.ts +40 -0
  195. package/src/components/delivery-notes/list/locales/is.ts +40 -0
  196. package/src/components/delivery-notes/list/locales/nb.ts +40 -0
  197. package/src/components/delivery-notes/list/locales/sk.ts +40 -0
  198. package/src/components/delivery-notes/list/locales/sv.ts +40 -0
  199. package/src/components/delivery-notes/list/use-delivery-note-download.ts +3 -5
  200. package/src/components/documents/create/document-add-item-form.tsx +256 -147
  201. package/src/components/documents/create/document-add-item-tax-rate-field.tsx +2 -1
  202. package/src/components/documents/create/document-details-section.tsx +30 -12
  203. package/src/components/documents/create/document-item-validation.ts +50 -0
  204. package/src/components/documents/create/document-items-section.tsx +44 -6
  205. package/src/components/documents/create/document-recipient-section.tsx +10 -2
  206. package/src/components/documents/create/linked-documents-info.tsx +4 -3
  207. package/src/components/documents/create/live-preview.tsx +19 -15
  208. package/src/components/documents/create/mark-as-paid-section.tsx +77 -20
  209. package/src/components/documents/create/payment-rows.ts +291 -0
  210. package/src/components/documents/create/prepare-document-submission.ts +24 -13
  211. package/src/components/documents/create/scroll-to-first-invalid-field.ts +27 -0
  212. package/src/components/documents/create/smart-code-insert-button.tsx +5 -53
  213. package/src/components/documents/create/use-document-customer-form.ts +74 -25
  214. package/src/components/documents/documents.hooks.ts +3 -5
  215. package/src/components/documents/index.ts +1 -0
  216. package/src/components/documents/public/index.ts +1 -0
  217. package/src/components/documents/public/public-document-summary.tsx +343 -0
  218. package/src/components/documents/shared/document-preview-display.tsx +65 -39
  219. package/src/components/documents/shared/index.ts +6 -1
  220. package/src/components/documents/shared/scaled-document-preview.tsx +33 -19
  221. package/src/components/documents/types.ts +24 -0
  222. package/src/components/documents/view/document-actions-bar.tsx +241 -62
  223. package/src/components/documents/view/document-activities-list.tsx +23 -11
  224. package/src/components/documents/view/document-details-card.tsx +7 -27
  225. package/src/components/documents/view/document-payments-list.tsx +114 -57
  226. package/src/components/documents/view/document-relations-list.tsx +26 -6
  227. package/src/components/documents/view/document-sidebar.tsx +25 -3
  228. package/src/components/documents/view/document-version-history.tsx +137 -0
  229. package/src/components/documents/view/index.ts +1 -0
  230. package/src/components/documents/view/locales/bg.ts +143 -0
  231. package/src/components/documents/view/locales/cs.ts +143 -0
  232. package/src/components/documents/view/locales/de.ts +13 -0
  233. package/src/components/documents/view/locales/en.ts +143 -0
  234. package/src/components/documents/view/locales/es.ts +13 -0
  235. package/src/components/documents/view/locales/et.ts +143 -0
  236. package/src/components/documents/view/locales/fi.ts +143 -0
  237. package/src/components/documents/view/locales/fr.ts +13 -0
  238. package/src/components/documents/view/locales/hr.ts +13 -0
  239. package/src/components/documents/view/locales/is.ts +143 -0
  240. package/src/components/documents/view/locales/it.ts +13 -0
  241. package/src/components/documents/view/locales/nb.ts +143 -0
  242. package/src/components/documents/view/locales/nl.ts +13 -0
  243. package/src/components/documents/view/locales/pl.ts +13 -0
  244. package/src/components/documents/view/locales/pt.ts +13 -0
  245. package/src/components/documents/view/locales/sk.ts +143 -0
  246. package/src/components/documents/view/locales/sl.ts +13 -0
  247. package/src/components/documents/view/locales/sv.ts +143 -0
  248. package/src/components/documents/view/use-document-download.ts +18 -17
  249. package/src/components/entities/create-entity-form.tsx +200 -12
  250. package/src/components/entities/entities.hooks.ts +15 -7
  251. package/src/components/entities/entity-settings-form/entity-settings-form.tsx +121 -65
  252. package/src/components/entities/entity-settings-form/input-with-preview.tsx +28 -4
  253. package/src/components/entities/entity-settings-form/locales/bg.ts +199 -0
  254. package/src/components/entities/entity-settings-form/locales/cs.ts +199 -0
  255. package/src/components/entities/entity-settings-form/locales/de.ts +8 -3
  256. package/src/components/entities/entity-settings-form/locales/en.ts +199 -0
  257. package/src/components/entities/entity-settings-form/locales/es.ts +8 -3
  258. package/src/components/entities/entity-settings-form/locales/et.ts +198 -0
  259. package/src/components/entities/entity-settings-form/locales/fi.ts +199 -0
  260. package/src/components/entities/entity-settings-form/locales/fr.ts +8 -3
  261. package/src/components/entities/entity-settings-form/locales/hr.ts +16 -11
  262. package/src/components/entities/entity-settings-form/locales/is.ts +199 -0
  263. package/src/components/entities/entity-settings-form/locales/it.ts +8 -3
  264. package/src/components/entities/entity-settings-form/locales/nb.ts +199 -0
  265. package/src/components/entities/entity-settings-form/locales/nl.ts +8 -3
  266. package/src/components/entities/entity-settings-form/locales/pl.ts +8 -3
  267. package/src/components/entities/entity-settings-form/locales/pt.ts +8 -3
  268. package/src/components/entities/entity-settings-form/locales/sk.ts +199 -0
  269. package/src/components/entities/entity-settings-form/locales/sl.ts +20 -15
  270. package/src/components/entities/entity-settings-form/locales/sv.ts +205 -0
  271. package/src/components/entities/fina-settings-form/fina-operator-required-dialog.tsx +34 -9
  272. package/src/components/entities/fina-settings-form/fina-settings-form.tsx +2 -0
  273. package/src/components/entities/fina-settings-form/fina-settings.hooks.ts +14 -45
  274. package/src/components/entities/fina-settings-form/locales/bg.ts +163 -0
  275. package/src/components/entities/fina-settings-form/locales/cs.ts +163 -0
  276. package/src/components/entities/fina-settings-form/locales/et.ts +163 -0
  277. package/src/components/entities/fina-settings-form/locales/fi.ts +163 -0
  278. package/src/components/entities/fina-settings-form/locales/hr.ts +11 -10
  279. package/src/components/entities/fina-settings-form/locales/is.ts +164 -0
  280. package/src/components/entities/fina-settings-form/locales/nb.ts +163 -0
  281. package/src/components/entities/fina-settings-form/locales/sk.ts +163 -0
  282. package/src/components/entities/fina-settings-form/locales/sl.ts +8 -7
  283. package/src/components/entities/fina-settings-form/locales/sv.ts +165 -0
  284. package/src/components/entities/furs-settings-form/furs-operator-required-dialog.tsx +34 -9
  285. package/src/components/entities/furs-settings-form/furs-settings-form.tsx +13 -1
  286. package/src/components/entities/furs-settings-form/furs-settings.hooks.ts +141 -42
  287. package/src/components/entities/furs-settings-form/locales/bg.ts +242 -0
  288. package/src/components/entities/furs-settings-form/locales/cs.ts +241 -0
  289. package/src/components/entities/furs-settings-form/locales/de.ts +42 -15
  290. package/src/components/entities/furs-settings-form/locales/en.ts +24 -0
  291. package/src/components/entities/furs-settings-form/locales/es.ts +42 -15
  292. package/src/components/entities/furs-settings-form/locales/et.ts +242 -0
  293. package/src/components/entities/furs-settings-form/locales/fi.ts +243 -0
  294. package/src/components/entities/furs-settings-form/locales/fr.ts +43 -15
  295. package/src/components/entities/furs-settings-form/locales/hr.ts +44 -19
  296. package/src/components/entities/furs-settings-form/locales/is.ts +242 -0
  297. package/src/components/entities/furs-settings-form/locales/it.ts +42 -15
  298. package/src/components/entities/furs-settings-form/locales/nb.ts +242 -0
  299. package/src/components/entities/furs-settings-form/locales/nl.ts +42 -15
  300. package/src/components/entities/furs-settings-form/locales/pl.ts +40 -15
  301. package/src/components/entities/furs-settings-form/locales/pt.ts +42 -15
  302. package/src/components/entities/furs-settings-form/locales/sk.ts +243 -0
  303. package/src/components/entities/furs-settings-form/locales/sl.ts +34 -10
  304. package/src/components/entities/furs-settings-form/locales/sv.ts +242 -0
  305. package/src/components/entities/furs-settings-form/sections/enable-fiscalization-section.tsx +11 -0
  306. package/src/components/entities/furs-settings-form/sections/furs-internal-act-download-card.tsx +111 -0
  307. package/src/components/entities/furs-settings-form/sections/general-settings-section.tsx +27 -0
  308. package/src/components/entities/furs-settings-form/sections/premises-management-section.tsx +23 -16
  309. package/src/components/entities/furs-settings-form/sections/register-premise-dialog.tsx +89 -31
  310. package/src/components/entities/settings/branding-settings-form.tsx +11 -30
  311. package/src/components/entities/settings/company-settings-form.tsx +2 -1
  312. package/src/components/entities/settings/defaults-settings-form.tsx +106 -3
  313. package/src/components/entities/settings/email-settings-form.tsx +245 -3
  314. package/src/components/entities/settings/eslog-settings-form.tsx +2 -1
  315. package/src/components/entities/settings/number-format-settings-form.tsx +2 -1
  316. package/src/components/entities/settings/pdf-template-selector/locales/bg.ts +20 -0
  317. package/src/components/entities/settings/pdf-template-selector/locales/cs.ts +20 -0
  318. package/src/components/entities/settings/pdf-template-selector/locales/en.ts +20 -0
  319. package/src/components/entities/settings/pdf-template-selector/locales/et.ts +19 -0
  320. package/src/components/entities/settings/pdf-template-selector/locales/fi.ts +20 -0
  321. package/src/components/entities/settings/pdf-template-selector/locales/is.ts +19 -0
  322. package/src/components/entities/settings/pdf-template-selector/locales/nb.ts +19 -0
  323. package/src/components/entities/settings/pdf-template-selector/locales/sk.ts +20 -0
  324. package/src/components/entities/settings/pdf-template-selector/locales/sv.ts +19 -0
  325. package/src/components/entities/settings/tax-rules-settings-form.tsx +21 -6
  326. package/src/components/estimates/create/create-estimate-form.tsx +70 -34
  327. package/src/components/estimates/create/locales/bg.ts +76 -0
  328. package/src/components/estimates/create/locales/cs.ts +76 -0
  329. package/src/components/estimates/create/locales/en.ts +76 -0
  330. package/src/components/estimates/create/locales/et.ts +76 -0
  331. package/src/components/estimates/create/locales/fi.ts +76 -0
  332. package/src/components/estimates/create/locales/hr.ts +8 -8
  333. package/src/components/estimates/create/locales/is.ts +76 -0
  334. package/src/components/estimates/create/locales/nb.ts +76 -0
  335. package/src/components/estimates/create/locales/sk.ts +76 -0
  336. package/src/components/estimates/create/locales/sl.ts +8 -8
  337. package/src/components/estimates/create/locales/sv.ts +76 -0
  338. package/src/components/estimates/create/prepare-estimate-submission.ts +4 -1
  339. package/src/components/estimates/estimates.hooks.ts +15 -3
  340. package/src/components/estimates/list/list-row-actions.tsx +34 -7
  341. package/src/components/estimates/list/list-table.tsx +78 -10
  342. package/src/components/estimates/list/locales/bg.ts +38 -0
  343. package/src/components/estimates/list/locales/cs.ts +38 -0
  344. package/src/components/estimates/list/locales/en.ts +27 -2
  345. package/src/components/estimates/list/locales/et.ts +38 -0
  346. package/src/components/estimates/list/locales/fi.ts +38 -0
  347. package/src/components/estimates/list/locales/hr.ts +2 -2
  348. package/src/components/estimates/list/locales/is.ts +38 -0
  349. package/src/components/estimates/list/locales/nb.ts +38 -0
  350. package/src/components/estimates/list/locales/sk.ts +38 -0
  351. package/src/components/estimates/list/locales/sv.ts +38 -0
  352. package/src/components/estimates/list/use-estimate-download.ts +3 -5
  353. package/src/components/export/document-export-form.tsx +72 -15
  354. package/src/components/export/sales-per-item-export-form.tsx +3 -2
  355. package/src/components/invoices/create/create-invoice-form.tsx +294 -76
  356. package/src/components/invoices/create/eslog-validation.ts +97 -8
  357. package/src/components/invoices/create/locales/bg.ts +164 -0
  358. package/src/components/invoices/create/locales/cs.ts +164 -0
  359. package/src/components/invoices/create/locales/de.ts +24 -0
  360. package/src/components/invoices/create/locales/en.ts +164 -0
  361. package/src/components/invoices/create/locales/es.ts +24 -0
  362. package/src/components/invoices/create/locales/et.ts +164 -0
  363. package/src/components/invoices/create/locales/fi.ts +164 -0
  364. package/src/components/invoices/create/locales/fr.ts +23 -0
  365. package/src/components/invoices/create/locales/hr.ts +32 -9
  366. package/src/components/invoices/create/locales/is.ts +164 -0
  367. package/src/components/invoices/create/locales/it.ts +24 -0
  368. package/src/components/invoices/create/locales/nb.ts +164 -0
  369. package/src/components/invoices/create/locales/nl.ts +23 -0
  370. package/src/components/invoices/create/locales/pl.ts +23 -0
  371. package/src/components/invoices/create/locales/pt.ts +24 -0
  372. package/src/components/invoices/create/locales/sk.ts +164 -0
  373. package/src/components/invoices/create/locales/sl.ts +32 -9
  374. package/src/components/invoices/create/locales/sv.ts +166 -0
  375. package/src/components/invoices/create/prepare-invoice-submission.ts +19 -2
  376. package/src/components/invoices/invoices-furs.hooks.ts +11 -8
  377. package/src/components/invoices/invoices.hooks.ts +27 -10
  378. package/src/components/invoices/list/list-row-actions.tsx +54 -21
  379. package/src/components/invoices/list/list-table.tsx +92 -11
  380. package/src/components/invoices/list/locales/bg.ts +49 -0
  381. package/src/components/invoices/list/locales/cs.ts +49 -0
  382. package/src/components/invoices/list/locales/en.ts +34 -2
  383. package/src/components/invoices/list/locales/et.ts +49 -0
  384. package/src/components/invoices/list/locales/fi.ts +49 -0
  385. package/src/components/invoices/list/locales/is.ts +49 -0
  386. package/src/components/invoices/list/locales/nb.ts +49 -0
  387. package/src/components/invoices/list/locales/sk.ts +49 -0
  388. package/src/components/invoices/list/locales/sv.ts +49 -0
  389. package/src/components/invoices/list/use-invoice-download.ts +3 -5
  390. package/src/components/invoices/send-email-dialog/locales/bg.ts +58 -0
  391. package/src/components/invoices/send-email-dialog/locales/cs.ts +58 -0
  392. package/src/components/invoices/send-email-dialog/locales/de.ts +21 -0
  393. package/src/components/invoices/send-email-dialog/locales/en.ts +58 -0
  394. package/src/components/invoices/send-email-dialog/locales/es.ts +21 -0
  395. package/src/components/invoices/send-email-dialog/locales/et.ts +58 -0
  396. package/src/components/invoices/send-email-dialog/locales/fi.ts +58 -0
  397. package/src/components/invoices/send-email-dialog/locales/fr.ts +21 -0
  398. package/src/components/invoices/send-email-dialog/locales/hr.ts +21 -0
  399. package/src/components/invoices/send-email-dialog/locales/is.ts +58 -0
  400. package/src/components/invoices/send-email-dialog/locales/it.ts +21 -0
  401. package/src/components/invoices/send-email-dialog/locales/nb.ts +58 -0
  402. package/src/components/invoices/send-email-dialog/locales/nl.ts +21 -0
  403. package/src/components/invoices/send-email-dialog/locales/pl.ts +21 -0
  404. package/src/components/invoices/send-email-dialog/locales/pt.ts +21 -0
  405. package/src/components/invoices/send-email-dialog/locales/sk.ts +58 -0
  406. package/src/components/invoices/send-email-dialog/locales/sl.ts +21 -0
  407. package/src/components/invoices/send-email-dialog/locales/sv.ts +58 -0
  408. package/src/components/invoices/send-email-dialog/send-email-dialog.tsx +188 -93
  409. package/src/components/invoices/view/eslog-info-display.tsx +2 -0
  410. package/src/components/invoices/view/fiscalization-status-card.tsx +2 -0
  411. package/src/components/items/create-item-form/create-item-form.tsx +29 -0
  412. package/src/components/items/create-item-form/locales/bg.ts +13 -0
  413. package/src/components/items/create-item-form/locales/cs.ts +13 -0
  414. package/src/components/items/create-item-form/locales/et.ts +13 -0
  415. package/src/components/items/create-item-form/locales/fi.ts +13 -0
  416. package/src/components/items/create-item-form/locales/is.ts +13 -0
  417. package/src/components/items/create-item-form/locales/nb.ts +13 -0
  418. package/src/components/items/create-item-form/locales/sk.ts +13 -0
  419. package/src/components/items/create-item-form/locales/sv.ts +13 -0
  420. package/src/components/items/edit-item-form/edit-item-form.tsx +39 -1
  421. package/src/components/items/edit-item-form/locales/bg.ts +12 -0
  422. package/src/components/items/edit-item-form/locales/cs.ts +12 -0
  423. package/src/components/items/edit-item-form/locales/en.ts +12 -0
  424. package/src/components/items/edit-item-form/locales/et.ts +12 -0
  425. package/src/components/items/edit-item-form/locales/fi.ts +12 -0
  426. package/src/components/items/edit-item-form/locales/is.ts +12 -0
  427. package/src/components/items/edit-item-form/locales/nb.ts +12 -0
  428. package/src/components/items/edit-item-form/locales/sk.ts +12 -0
  429. package/src/components/items/edit-item-form/locales/sv.ts +12 -0
  430. package/src/components/items/item-combobox.tsx +51 -18
  431. package/src/components/items/item-list-table/item-list-table.tsx +47 -23
  432. package/src/components/items/item-list-table/locales/bg.ts +16 -0
  433. package/src/components/items/item-list-table/locales/cs.ts +16 -0
  434. package/src/components/items/item-list-table/locales/en.ts +4 -3
  435. package/src/components/items/item-list-table/locales/et.ts +16 -0
  436. package/src/components/items/item-list-table/locales/fi.ts +16 -0
  437. package/src/components/items/item-list-table/locales/is.ts +16 -0
  438. package/src/components/items/item-list-table/locales/nb.ts +16 -0
  439. package/src/components/items/item-list-table/locales/sk.ts +16 -0
  440. package/src/components/items/item-list-table/locales/sv.ts +16 -0
  441. package/src/components/items/items.hooks.ts +15 -15
  442. package/src/components/payments/create-payment-form/create-payment-form.tsx +11 -7
  443. package/src/components/payments/create-payment-form/locales/bg.ts +20 -0
  444. package/src/components/payments/create-payment-form/locales/cs.ts +20 -0
  445. package/src/components/payments/create-payment-form/locales/en.ts +20 -0
  446. package/src/components/payments/create-payment-form/locales/et.ts +20 -0
  447. package/src/components/payments/create-payment-form/locales/fi.ts +20 -0
  448. package/src/components/payments/create-payment-form/locales/hr.ts +2 -2
  449. package/src/components/payments/create-payment-form/locales/is.ts +20 -0
  450. package/src/components/payments/create-payment-form/locales/nb.ts +20 -0
  451. package/src/components/payments/create-payment-form/locales/sk.ts +20 -0
  452. package/src/components/payments/create-payment-form/locales/sl.ts +4 -4
  453. package/src/components/payments/create-payment-form/locales/sv.ts +20 -0
  454. package/src/components/payments/edit-payment-form/edit-payment-form.tsx +5 -1
  455. package/src/components/payments/edit-payment-form/locales/bg.ts +20 -0
  456. package/src/components/payments/edit-payment-form/locales/cs.ts +20 -0
  457. package/src/components/payments/edit-payment-form/locales/en.ts +20 -0
  458. package/src/components/payments/edit-payment-form/locales/et.ts +20 -0
  459. package/src/components/payments/edit-payment-form/locales/fi.ts +20 -0
  460. package/src/components/payments/edit-payment-form/locales/hr.ts +2 -2
  461. package/src/components/payments/edit-payment-form/locales/is.ts +20 -0
  462. package/src/components/payments/edit-payment-form/locales/nb.ts +20 -0
  463. package/src/components/payments/edit-payment-form/locales/sk.ts +20 -0
  464. package/src/components/payments/edit-payment-form/locales/sl.ts +4 -4
  465. package/src/components/payments/edit-payment-form/locales/sv.ts +20 -0
  466. package/src/components/payments/list/list-row-actions.tsx +10 -2
  467. package/src/components/payments/list/list-table.tsx +51 -45
  468. package/src/components/payments/list/locales/bg.ts +42 -0
  469. package/src/components/payments/list/locales/cs.ts +42 -0
  470. package/src/components/payments/list/locales/de.ts +18 -0
  471. package/src/components/payments/list/locales/en.ts +37 -2
  472. package/src/components/payments/list/locales/es.ts +18 -0
  473. package/src/components/payments/list/locales/et.ts +42 -0
  474. package/src/components/payments/list/locales/fi.ts +42 -0
  475. package/src/components/payments/list/locales/fr.ts +18 -0
  476. package/src/components/payments/list/locales/hr.ts +18 -0
  477. package/src/components/payments/list/locales/is.ts +42 -0
  478. package/src/components/payments/list/locales/it.ts +18 -0
  479. package/src/components/payments/list/locales/nb.ts +42 -0
  480. package/src/components/payments/list/locales/nl.ts +18 -0
  481. package/src/components/payments/list/locales/pl.ts +18 -0
  482. package/src/components/payments/list/locales/pt.ts +18 -0
  483. package/src/components/payments/list/locales/sk.ts +42 -0
  484. package/src/components/payments/list/locales/sl.ts +18 -1
  485. package/src/components/payments/list/locales/sv.ts +42 -0
  486. package/src/components/payments/payments.hooks.ts +13 -6
  487. package/src/components/recurring-invoices/create-recurring-invoice-form/create-recurring-invoice-form.tsx +23 -17
  488. package/src/components/recurring-invoices/create-recurring-invoice-form/locales/bg.ts +44 -0
  489. package/src/components/recurring-invoices/create-recurring-invoice-form/locales/cs.ts +44 -0
  490. package/src/components/recurring-invoices/create-recurring-invoice-form/locales/en.ts +44 -0
  491. package/src/components/recurring-invoices/create-recurring-invoice-form/locales/et.ts +44 -0
  492. package/src/components/recurring-invoices/create-recurring-invoice-form/locales/fi.ts +44 -0
  493. package/src/components/recurring-invoices/create-recurring-invoice-form/locales/hr.ts +6 -6
  494. package/src/components/recurring-invoices/create-recurring-invoice-form/locales/is.ts +44 -0
  495. package/src/components/recurring-invoices/create-recurring-invoice-form/locales/nb.ts +44 -0
  496. package/src/components/recurring-invoices/create-recurring-invoice-form/locales/sk.ts +44 -0
  497. package/src/components/recurring-invoices/create-recurring-invoice-form/locales/sl.ts +4 -4
  498. package/src/components/recurring-invoices/create-recurring-invoice-form/locales/sv.ts +44 -0
  499. package/src/components/recurring-invoices/list/list-row-actions.tsx +3 -5
  500. package/src/components/recurring-invoices/list/list-table.tsx +10 -7
  501. package/src/components/recurring-invoices/list/locales/bg.ts +28 -0
  502. package/src/components/recurring-invoices/list/locales/cs.ts +28 -0
  503. package/src/components/recurring-invoices/list/locales/en.ts +23 -0
  504. package/src/components/recurring-invoices/list/locales/et.ts +28 -0
  505. package/src/components/recurring-invoices/list/locales/fi.ts +28 -0
  506. package/src/components/recurring-invoices/list/locales/is.ts +28 -0
  507. package/src/components/recurring-invoices/list/locales/nb.ts +28 -0
  508. package/src/components/recurring-invoices/list/locales/sk.ts +28 -0
  509. package/src/components/recurring-invoices/list/locales/sv.ts +28 -0
  510. package/src/components/recurring-invoices/recurring-invoices.hooks.ts +11 -8
  511. package/src/components/request-logs/request-log-detail.tsx +30 -23
  512. package/src/components/request-logs/request-log-list-table.tsx +88 -45
  513. package/src/components/table/README.md +47 -9
  514. package/src/components/table/data-table.tsx +232 -92
  515. package/src/components/table/filter-bar.tsx +1 -1
  516. package/src/components/table/filter-panel.tsx +51 -22
  517. package/src/components/table/hooks/use-table-state.ts +64 -11
  518. package/src/components/table/index.ts +3 -0
  519. package/src/components/table/locales.ts +119 -4
  520. package/src/components/table/selection-toolbar.tsx +65 -9
  521. package/src/components/table/sorting.ts +80 -0
  522. package/src/components/table/table-skeleton.tsx +15 -9
  523. package/src/components/table/types.ts +17 -0
  524. package/src/components/tax-reports/index.ts +8 -0
  525. package/src/components/tax-reports/kir-export-form.tsx +4 -5
  526. package/src/components/tax-reports/slovenia-accounting-mappings-fields.tsx +271 -0
  527. package/src/components/tax-reports/slovenia-tax-profile-step.tsx +21 -10
  528. package/src/components/tax-reports/slovenia-vod-export-form.tsx +452 -0
  529. package/src/components/tax-reports/slovenia-yearly-export-form.tsx +17 -16
  530. package/src/components/tax-reports/slovenia-yearly-review-step.tsx +40 -18
  531. package/src/components/taxes/create-tax-form/create-tax-form.tsx +48 -2
  532. package/src/components/taxes/create-tax-form/locales/bg.ts +8 -0
  533. package/src/components/taxes/create-tax-form/locales/cs.ts +8 -0
  534. package/src/components/taxes/create-tax-form/locales/et.ts +8 -0
  535. package/src/components/taxes/create-tax-form/locales/fi.ts +8 -0
  536. package/src/components/taxes/create-tax-form/locales/is.ts +8 -0
  537. package/src/components/taxes/create-tax-form/locales/nb.ts +8 -0
  538. package/src/components/taxes/create-tax-form/locales/sk.ts +8 -0
  539. package/src/components/taxes/create-tax-form/locales/sv.ts +8 -0
  540. package/src/components/taxes/edit-tax-form/edit-tax-form.tsx +80 -11
  541. package/src/components/taxes/edit-tax-form/locales/bg.ts +6 -0
  542. package/src/components/taxes/edit-tax-form/locales/cs.ts +6 -0
  543. package/src/components/taxes/edit-tax-form/locales/en.ts +6 -0
  544. package/src/components/taxes/edit-tax-form/locales/et.ts +6 -0
  545. package/src/components/taxes/edit-tax-form/locales/fi.ts +6 -0
  546. package/src/components/taxes/edit-tax-form/locales/is.ts +6 -0
  547. package/src/components/taxes/edit-tax-form/locales/nb.ts +6 -0
  548. package/src/components/taxes/edit-tax-form/locales/sk.ts +6 -0
  549. package/src/components/taxes/edit-tax-form/locales/sv.ts +6 -0
  550. package/src/components/taxes/tax-list-table/locales/bg.ts +17 -0
  551. package/src/components/taxes/tax-list-table/locales/cs.ts +17 -0
  552. package/src/components/taxes/tax-list-table/locales/en.ts +4 -3
  553. package/src/components/taxes/tax-list-table/locales/et.ts +17 -0
  554. package/src/components/taxes/tax-list-table/locales/fi.ts +17 -0
  555. package/src/components/taxes/tax-list-table/locales/is.ts +17 -0
  556. package/src/components/taxes/tax-list-table/locales/nb.ts +17 -0
  557. package/src/components/taxes/tax-list-table/locales/sk.ts +17 -0
  558. package/src/components/taxes/tax-list-table/locales/sv.ts +17 -0
  559. package/src/components/taxes/tax-list-table/tax-list-table.tsx +53 -19
  560. package/src/components/taxes/taxes.hooks.ts +53 -12
  561. package/src/components/ui/badge.tsx +1 -1
  562. package/src/components/ui/button.tsx +1 -1
  563. package/src/components/ui/drawer.tsx +2 -2
  564. package/src/components/ui/form.tsx +118 -2
  565. package/src/components/ui/progress.tsx +6 -4
  566. package/src/components/ui/select.tsx +118 -1
  567. package/src/components/ui/sonner.tsx +17 -36
  568. package/src/components/ui/sticky-form-footer.tsx +6 -1
  569. package/src/components/ui/tooltip.tsx +15 -2
  570. package/src/components/webhook-logs/webhook-delivery-detail.tsx +22 -22
  571. package/src/components/webhook-logs/webhook-delivery-list-table.tsx +92 -17
  572. package/src/components/wl-subscription/locked-feature.tsx +11 -10
  573. package/src/components/wl-subscription/paywall.tsx +184 -72
  574. package/src/components/wl-subscription/upgrade-modal.tsx +28 -31
  575. package/src/generate-schemas.ts +14 -4
  576. package/src/generated/schemas/account_webhook.ts +43 -0
  577. package/src/generated/schemas/accountwebhook.ts +67 -0
  578. package/src/generated/schemas/entity.ts +255 -16
  579. package/src/generated/schemas/exportsloveniavodxml_body.ts +16 -0
  580. package/src/generated/schemas/furssettings.ts +1 -0
  581. package/src/generated/schemas/incomingpurchasedocument.ts +198 -0
  582. package/src/generated/schemas/incomingpurchasedocumentpayment.ts +38 -0
  583. package/src/generated/schemas/index.ts +10 -0
  584. package/src/generated/schemas/item.ts +36 -0
  585. package/src/generated/schemas/me.ts +23 -19
  586. package/src/generated/schemas/order.ts +2 -0
  587. package/src/generated/schemas/orderintegration.ts +8 -4
  588. package/src/generated/schemas/payment.ts +5 -0
  589. package/src/generated/schemas/ptatcudseries.ts +30 -0
  590. package/src/generated/schemas/renderadvanceinvoicepreview_body.ts +35 -3
  591. package/src/generated/schemas/rendercreditnotepreview_body.ts +35 -3
  592. package/src/generated/schemas/renderdeliverynotepreview_body.ts +12 -5
  593. package/src/generated/schemas/renderestimatepreview_body.ts +38 -5
  594. package/src/generated/schemas/renderinvoicepreview_body.ts +39 -5
  595. package/src/generated/schemas/senddocument_body.ts +1 -0
  596. package/src/generated/schemas/sendemail_body.ts +67 -39
  597. package/src/generated/schemas/sloveniataxprofile.ts +97 -21
  598. package/src/generated/schemas/starteslogexport_body.ts +33 -0
  599. package/src/generated/schemas/startpdfexport_body.ts +94 -78
  600. package/src/generated/schemas/supplier.ts +59 -0
  601. package/src/generated/schemas/syncshopifyorders_body.ts +20 -0
  602. package/src/generated/schemas/tax.ts +2 -0
  603. package/src/generated/schemas/uploadfile_body.ts +1 -0
  604. package/src/generated/schemas/userptsettings.ts +26 -0
  605. package/src/generated/schemas/voidinvoice_body.ts +20 -0
  606. package/src/generated/schemas/webhook.ts +2 -0
  607. package/src/hooks/create-resource-hooks.ts +22 -19
  608. package/src/hooks/use-duplicate-document.ts +10 -8
  609. package/src/hooks/use-eslog-validation.ts +5 -1
  610. package/src/hooks/use-next-document-number.ts +13 -6
  611. package/src/hooks/use-resolved-entity-id.ts +14 -0
  612. package/src/hooks/use-resource-mutation.ts +11 -30
  613. package/src/hooks/use-transaction-type-check.ts +3 -8
  614. package/src/lib/browser-cookies.ts +1 -1
  615. package/src/lib/country-capabilities.ts +114 -0
  616. package/src/lib/date-fns-locale.ts +1 -20
  617. package/src/lib/eslog-export.ts +32 -0
  618. package/src/lib/fiscalization-options.ts +21 -1
  619. package/src/lib/formatting.ts +27 -0
  620. package/src/lib/furs-error-utils.ts +29 -0
  621. package/src/lib/locale.ts +4 -0
  622. package/src/lib/payment-display.ts +95 -0
  623. package/src/lib/pt-document-input.ts +67 -0
  624. package/src/lib/schemas/shared.ts +1 -0
  625. package/src/lib/template-variables.tsx +102 -4
  626. package/src/lib/translation.ts +51 -33
  627. package/src/lib/white-label-capabilities.ts +626 -0
  628. package/src/lib/zod-validation-message.ts +291 -0
  629. package/src/providers/entities-provider.tsx +32 -14
  630. package/src/providers/sdk-provider.tsx +112 -12
  631. package/src/providers/space-invoices-provider.tsx +129 -0
  632. package/src/providers/white-label-provider.tsx +96 -4
  633. package/src/providers/wl-subscription-provider.tsx +144 -10
@@ -1,17 +1,22 @@
1
+ import { ArrowDown, ArrowUp, ArrowUpDown } from "lucide-react";
1
2
  import type { JSX, ReactNode } from "react";
2
- import { Fragment, memo, useCallback, useMemo, useState } from "react";
3
+ import { Fragment, lazy, memo, Suspense, useCallback, useMemo, useState } from "react";
3
4
 
4
5
  import { Checkbox } from "@/ui/components/ui/checkbox";
5
6
  import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow } from "@/ui/components/ui/table";
6
- import { FilterBar } from "./filter-bar";
7
+ import { cn } from "@/ui/lib/utils";
7
8
  import { useTableQuery } from "./hooks/use-table-query";
8
9
  import { useTableState } from "./hooks/use-table-state";
10
+ import { SearchInput } from "./search-input";
11
+ import { getColumnSortDirection, getNextColumnOrderBy, isSortableColumn } from "./sorting";
9
12
  import { TableEmptyState } from "./table-empty-state";
10
13
  import { TableNoResults } from "./table-no-results";
11
14
  import { Pagination } from "./table-pagination";
12
15
  import { TableSkeleton } from "./table-skeleton";
13
16
  import type { Column, FilterConfig, TableQueryParams, TableQueryResponse } from "./types";
14
17
 
18
+ const LazyFilterBar = lazy(() => import("./filter-bar").then((module) => ({ default: module.FilterBar })));
19
+
15
20
  export type DataTableProps<T> = {
16
21
  /** Column definitions */
17
22
  columns: Column<T>[];
@@ -55,8 +60,49 @@ export type DataTableProps<T> = {
55
60
  onSelectionChange?: (selectedIds: Set<string>) => void;
56
61
  /** Content to render in selection toolbar (shown when items selected) */
57
62
  selectionToolbar?: (selectedCount: number, data: T[]) => ReactNode;
63
+ /** When false, hides the search / filter toolbar area */
64
+ showSearchToolbar?: boolean;
65
+ /** Custom empty state for truly empty collections */
66
+ emptyState?: ReactNode;
67
+ /** When false, hides the pagination footer */
68
+ showPagination?: boolean;
69
+ /** Horizontal inset around the table block and related states */
70
+ contentInsetClassName?: string;
71
+ /** Bottom padding on the overall table wrapper */
72
+ bottomPaddingClassName?: string;
58
73
  };
59
74
 
75
+ function hasFilterControls(filterConfig?: FilterConfig) {
76
+ return Boolean(
77
+ filterConfig?.dateFields?.length ||
78
+ filterConfig?.statusFilter ||
79
+ filterConfig?.httpMethodFilter ||
80
+ filterConfig?.httpStatusCodeFilter,
81
+ );
82
+ }
83
+
84
+ function SearchToolbar({
85
+ searchValue,
86
+ onSearch,
87
+ t,
88
+ }: {
89
+ searchValue?: string;
90
+ onSearch: (value: string | null) => void;
91
+ t: (key: string) => string;
92
+ }) {
93
+ return (
94
+ <div className="flex flex-col gap-2 px-4 pt-4 sm:flex-row sm:items-center">
95
+ <SearchInput
96
+ initialValue={searchValue}
97
+ onSearch={onSearch}
98
+ placeholder={t("Search...")}
99
+ ariaLabel={t("Search")}
100
+ clearAriaLabel={t("Clear search")}
101
+ />
102
+ </div>
103
+ );
104
+ }
105
+
60
106
  /**
61
107
  * Generic data table with built-in sorting, search, pagination, and loading states
62
108
  */
@@ -82,7 +128,14 @@ export function DataTable<T extends { id: string }>({
82
128
  selectedIds,
83
129
  onSelectionChange,
84
130
  selectionToolbar,
131
+ showSearchToolbar = true,
132
+ emptyState,
133
+ showPagination = true,
134
+ contentInsetClassName = "px-4",
135
+ bottomPaddingClassName = "pb-4",
85
136
  }: DataTableProps<T>) {
137
+ const hasFilters = hasFilterControls(filterConfig);
138
+ const displayRows = Math.max(queryParams?.limit ?? 10, 1);
86
139
  // Filter panel open state - starts open if filters are active in URL
87
140
  const hasInitialFilters = Boolean(
88
141
  queryParams?.filter_date_from ||
@@ -95,11 +148,12 @@ export function DataTable<T extends { id: string }>({
95
148
  const [filterPanelOpen, setFilterPanelOpen] = useState(hasInitialFilters);
96
149
 
97
150
  // Manage table state (search, pagination, filters)
98
- const { params, apiParams, filterState, handleSearch, handlePageChange, handleFilterChange } = useTableState({
99
- initialParams: queryParams,
100
- onChangeParams,
101
- disableUrlSync,
102
- });
151
+ const { params, apiParams, filterState, handleSearch, handlePageChange, handleFilterChange, handleSortChange } =
152
+ useTableState({
153
+ initialParams: queryParams,
154
+ onChangeParams,
155
+ disableUrlSync,
156
+ });
103
157
 
104
158
  // Fetch table data (use apiParams which has the query JSON for API)
105
159
  const { data: queryResult, isFetching } = useTableQuery<T>({
@@ -166,19 +220,33 @@ export function DataTable<T extends { id: string }>({
166
220
  // Show skeleton during initial load (with filter bar for consistency)
167
221
  if (isFetching && !queryResult) {
168
222
  return (
169
- <div className="space-y-4">
170
- <FilterBar
171
- searchValue={params.search}
172
- onSearch={handleSearch}
173
- filterConfig={filterConfig}
174
- filterState={filterState ?? undefined}
175
- onFilterChange={handleFilterChange}
176
- t={t}
177
- locale={locale}
178
- isOpen={filterPanelOpen}
179
- onToggle={setFilterPanelOpen}
180
- />
181
- <TableSkeleton columns={columns.length + (selectable ? 1 : 0)} rows={10} />
223
+ <div className={cn("space-y-4", bottomPaddingClassName)}>
224
+ {showSearchToolbar &&
225
+ (hasFilters ? (
226
+ <Suspense fallback={<SearchToolbar searchValue={params.search} onSearch={handleSearch} t={t} />}>
227
+ <LazyFilterBar
228
+ searchValue={params.search}
229
+ onSearch={handleSearch}
230
+ filterConfig={filterConfig}
231
+ filterState={filterState ?? undefined}
232
+ onFilterChange={handleFilterChange}
233
+ t={t}
234
+ locale={locale}
235
+ isOpen={filterPanelOpen}
236
+ onToggle={setFilterPanelOpen}
237
+ />
238
+ </Suspense>
239
+ ) : (
240
+ <SearchToolbar searchValue={params.search} onSearch={handleSearch} t={t} />
241
+ ))}
242
+ <div className={contentInsetClassName}>
243
+ <TableSkeleton
244
+ columns={columns.length + (selectable ? 1 : 0)}
245
+ rows={displayRows}
246
+ showSearch={false}
247
+ showPagination={showPagination}
248
+ />
249
+ </div>
182
250
  </div>
183
251
  );
184
252
  }
@@ -187,84 +255,107 @@ export function DataTable<T extends { id: string }>({
187
255
  // (this means truly empty collection, not filtered to zero results)
188
256
  if (data.length === 0 && !hasActiveFilters) {
189
257
  return (
190
- <TableEmptyState
191
- resource={resourceName}
192
- createNewLink={createNewLink}
193
- createNewTrigger={createNewTrigger}
194
- onCreateNew={onCreateNew}
195
- t={t}
196
- />
258
+ <div className={cn(contentInsetClassName, bottomPaddingClassName)}>
259
+ {emptyState ? (
260
+ emptyState
261
+ ) : (
262
+ <TableEmptyState
263
+ resource={resourceName}
264
+ createNewLink={createNewLink}
265
+ createNewTrigger={createNewTrigger}
266
+ onCreateNew={onCreateNew}
267
+ rows={displayRows}
268
+ t={t}
269
+ />
270
+ )}
271
+ </div>
197
272
  );
198
273
  }
199
274
 
200
275
  return (
201
- <div className="space-y-4">
202
- <FilterBar
203
- searchValue={params.search}
204
- onSearch={handleSearch}
205
- filterConfig={filterConfig}
206
- filterState={filterState ?? undefined}
207
- onFilterChange={handleFilterChange}
208
- t={t}
209
- locale={locale}
210
- isOpen={filterPanelOpen}
211
- onToggle={setFilterPanelOpen}
212
- />
276
+ <div className={cn("space-y-4", bottomPaddingClassName)}>
277
+ {showSearchToolbar &&
278
+ (hasFilters ? (
279
+ <Suspense fallback={<SearchToolbar searchValue={params.search} onSearch={handleSearch} t={t} />}>
280
+ <LazyFilterBar
281
+ searchValue={params.search}
282
+ onSearch={handleSearch}
283
+ filterConfig={filterConfig}
284
+ filterState={filterState ?? undefined}
285
+ onFilterChange={handleFilterChange}
286
+ t={t}
287
+ locale={locale}
288
+ isOpen={filterPanelOpen}
289
+ onToggle={setFilterPanelOpen}
290
+ />
291
+ </Suspense>
292
+ ) : (
293
+ <SearchToolbar searchValue={params.search} onSearch={handleSearch} t={t} />
294
+ ))}
213
295
 
214
296
  {selectable && selectedCount > 0 && selectionToolbar && (
215
- <div className="flex items-center gap-3 rounded-lg border bg-muted/50 px-4 py-2">
216
- {selectionToolbar(selectedCount, data)}
297
+ <div className={contentInsetClassName}>
298
+ <div className="flex items-center gap-3 rounded-lg border bg-muted/50 px-4 py-2">
299
+ {selectionToolbar(selectedCount, data)}
300
+ </div>
217
301
  </div>
218
302
  )}
219
303
 
220
- <div className="rounded-lg border">
221
- <Table>
222
- {renderHeader ? (
223
- renderHeader()
224
- ) : (
225
- <DefaultTableHeader
226
- columns={columns}
227
- selectable={selectable}
228
- allPageSelected={allPageSelected}
229
- somePageSelected={somePageSelected}
230
- onToggleAll={handleToggleAll}
231
- />
232
- )}
233
-
234
- <TableBody>
235
- {data.length > 0 ? (
236
- data.map((item) => {
237
- if (renderRow) {
238
- // Custom row renderer - wrap in Fragment with key
239
- return <Fragment key={item.id}>{renderRow(item)}</Fragment>;
240
- }
241
-
242
- // Default row renderer
243
- return (
244
- <DefaultTableRow
245
- key={item.id}
246
- item={item}
247
- columns={columns}
248
- onRowClick={onRowClick}
249
- selectable={selectable}
250
- isSelected={selectedIds?.has(item.id)}
251
- onToggleSelect={handleToggleRow}
252
- />
253
- );
254
- })
304
+ <div className={contentInsetClassName}>
305
+ <div className="rounded-lg border">
306
+ <Table>
307
+ {renderHeader ? (
308
+ renderHeader()
255
309
  ) : (
256
- <TableNoResults resource={resourceName} search={handleSearch} onClear={handleClearAll} t={t} />
310
+ <DefaultTableHeader
311
+ columns={columns}
312
+ selectable={selectable}
313
+ allPageSelected={allPageSelected}
314
+ somePageSelected={somePageSelected}
315
+ onToggleAll={handleToggleAll}
316
+ orderBy={params.order_by}
317
+ onSortChange={handleSortChange}
318
+ t={t}
319
+ />
257
320
  )}
258
- </TableBody>
259
- </Table>
260
321
 
261
- <div className="border-t px-4 py-3">
262
- <Pagination
263
- prevCursor={queryResult?.pagination.prev_cursor}
264
- nextCursor={queryResult?.pagination.next_cursor}
265
- onPageChange={handlePageChange}
266
- t={t}
267
- />
322
+ <TableBody>
323
+ {data.length > 0 ? (
324
+ data.map((item) => {
325
+ if (renderRow) {
326
+ // Custom row renderer - wrap in Fragment with key
327
+ return <Fragment key={item.id}>{renderRow(item)}</Fragment>;
328
+ }
329
+
330
+ // Default row renderer
331
+ return (
332
+ <DefaultTableRow
333
+ key={item.id}
334
+ item={item}
335
+ columns={columns}
336
+ onRowClick={onRowClick}
337
+ selectable={selectable}
338
+ isSelected={selectedIds?.has(item.id)}
339
+ onToggleSelect={handleToggleRow}
340
+ />
341
+ );
342
+ })
343
+ ) : (
344
+ <TableNoResults resource={resourceName} search={handleSearch} onClear={handleClearAll} t={t} />
345
+ )}
346
+ </TableBody>
347
+ </Table>
348
+
349
+ {showPagination && (
350
+ <div className="border-t px-4 py-3">
351
+ <Pagination
352
+ prevCursor={queryResult?.pagination.prev_cursor}
353
+ nextCursor={queryResult?.pagination.next_cursor}
354
+ onPageChange={handlePageChange}
355
+ t={t}
356
+ />
357
+ </div>
358
+ )}
268
359
  </div>
269
360
  </div>
270
361
  </div>
@@ -280,12 +371,18 @@ const DefaultTableHeader = memo(function DefaultTableHeader<T>({
280
371
  allPageSelected,
281
372
  somePageSelected,
282
373
  onToggleAll,
374
+ orderBy,
375
+ onSortChange,
376
+ t,
283
377
  }: {
284
378
  columns: Column<T>[];
285
379
  selectable?: boolean;
286
380
  allPageSelected?: boolean;
287
381
  somePageSelected?: boolean;
288
382
  onToggleAll?: () => void;
383
+ orderBy?: TableQueryParams["order_by"];
384
+ onSortChange?: (orderBy: TableQueryParams["order_by"]) => void;
385
+ t: (key: string) => string;
289
386
  }) {
290
387
  return (
291
388
  <TableHeader>
@@ -299,11 +396,47 @@ const DefaultTableHeader = memo(function DefaultTableHeader<T>({
299
396
  />
300
397
  </TableHead>
301
398
  )}
302
- {columns.map((column) => (
303
- <TableHead key={column.id} className={column.className} style={{ textAlign: column.align }}>
304
- {column.header}
305
- </TableHead>
306
- ))}
399
+ {columns.map((column) => {
400
+ const sortDirection = getColumnSortDirection(column, orderBy);
401
+ const sortable = isSortableColumn(column);
402
+ const SortIcon = sortDirection === "asc" ? ArrowUp : sortDirection === "desc" ? ArrowDown : ArrowUpDown;
403
+ const sortAriaLabel = typeof column.header === "string" ? `${t("Sort by")} ${column.header}` : t("Sort by");
404
+
405
+ return (
406
+ <TableHead
407
+ key={column.id}
408
+ className={cn(column.id === "actions" && "w-[52px] whitespace-nowrap", column.className)}
409
+ style={{ textAlign: column.align }}
410
+ aria-sort={
411
+ sortable
412
+ ? sortDirection === "asc"
413
+ ? "ascending"
414
+ : sortDirection === "desc"
415
+ ? "descending"
416
+ : "none"
417
+ : undefined
418
+ }
419
+ >
420
+ {sortable ? (
421
+ <button
422
+ type="button"
423
+ className={cn(
424
+ "flex w-full items-center gap-1 rounded py-1 text-left transition-colors hover:text-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring",
425
+ column.align === "center" && "justify-center",
426
+ column.align === "right" && "justify-end",
427
+ )}
428
+ onClick={() => onSortChange?.(getNextColumnOrderBy(column, orderBy))}
429
+ aria-label={sortAriaLabel}
430
+ >
431
+ <span>{column.header}</span>
432
+ <SortIcon className={cn("h-3.5 w-3.5 shrink-0", sortDirection === null && "text-muted-foreground")} />
433
+ </button>
434
+ ) : (
435
+ column.header
436
+ )}
437
+ </TableHead>
438
+ );
439
+ })}
307
440
  </TableRow>
308
441
  </TableHeader>
309
442
  );
@@ -313,6 +446,9 @@ const DefaultTableHeader = memo(function DefaultTableHeader<T>({
313
446
  allPageSelected?: boolean;
314
447
  somePageSelected?: boolean;
315
448
  onToggleAll?: () => void;
449
+ orderBy?: TableQueryParams["order_by"];
450
+ onSortChange?: (orderBy: TableQueryParams["order_by"]) => void;
451
+ t: (key: string) => string;
316
452
  }) => JSX.Element;
317
453
 
318
454
  /**
@@ -345,7 +481,11 @@ const DefaultTableRow = memo(function DefaultTableRow<T extends { id: string }>(
345
481
  </TableCell>
346
482
  )}
347
483
  {columns.map((column) => (
348
- <TableCell key={column.id} className={column.className} style={{ textAlign: column.align }}>
484
+ <TableCell
485
+ key={column.id}
486
+ className={cn(column.id === "actions" && "w-[52px] whitespace-nowrap", column.className)}
487
+ style={{ textAlign: column.align }}
488
+ >
349
489
  {column.cell?.(item)}
350
490
  </TableCell>
351
491
  ))}
@@ -53,7 +53,7 @@ export function FilterBar({
53
53
  if (!hasFilters) {
54
54
  // No filters configured, just show search
55
55
  return (
56
- <div className="px-4 pt-4">
56
+ <div className="flex flex-col gap-2 px-4 pt-4 sm:flex-row sm:items-center">
57
57
  <SearchInput
58
58
  initialValue={searchValue}
59
59
  onSearch={onSearch}
@@ -144,6 +144,8 @@ type DateRangeFilterProps = {
144
144
 
145
145
  function DateRangeFilter({ fields, value, onChange, t, locale }: DateRangeFilterProps) {
146
146
  const selectedField = value?.field ?? fields[0]?.id;
147
+ const selectedFieldLabel =
148
+ fields.find((field) => field.id === selectedField)?.label ?? fields[0]?.label ?? selectedField;
147
149
 
148
150
  const handleFieldChange = useCallback(
149
151
  (field: string) => {
@@ -192,7 +194,7 @@ function DateRangeFilter({ fields, value, onChange, t, locale }: DateRangeFilter
192
194
  {/* Field selector */}
193
195
  <Select value={selectedField} onValueChange={(v) => v && handleFieldChange(v)}>
194
196
  <SelectTrigger size="sm" className="w-[140px]">
195
- <SelectValue />
197
+ <SelectValue>{selectedFieldLabel}</SelectValue>
196
198
  </SelectTrigger>
197
199
  <SelectContent>
198
200
  {fields.map((field) => (
@@ -204,7 +206,13 @@ function DateRangeFilter({ fields, value, onChange, t, locale }: DateRangeFilter
204
206
  </Select>
205
207
 
206
208
  {/* From date */}
207
- <DatePicker value={value?.range.from} onChange={handleFromChange} placeholder={t("From")} locale={locale} />
209
+ <DatePicker
210
+ value={value?.range.from}
211
+ onChange={handleFromChange}
212
+ placeholder={t("From")}
213
+ t={t}
214
+ locale={locale}
215
+ />
208
216
 
209
217
  <span className="text-muted-foreground">–</span>
210
218
 
@@ -213,6 +221,7 @@ function DateRangeFilter({ fields, value, onChange, t, locale }: DateRangeFilter
213
221
  value={value?.range.to}
214
222
  onChange={handleToChange}
215
223
  placeholder={t("To")}
224
+ t={t}
216
225
  locale={locale}
217
226
  minDate={value?.range.from}
218
227
  />
@@ -225,11 +234,12 @@ type DatePickerProps = {
225
234
  value?: Date;
226
235
  onChange: (date: Date | undefined) => void;
227
236
  placeholder: string;
237
+ t: (key: string) => string;
228
238
  locale?: string;
229
239
  minDate?: Date;
230
240
  };
231
241
 
232
- function DatePicker({ value, onChange, placeholder, minDate }: DatePickerProps) {
242
+ function DatePicker({ value, onChange, placeholder, t, locale, minDate }: DatePickerProps) {
233
243
  const [open, setOpen] = useState(false);
234
244
 
235
245
  const handleSelect = useCallback(
@@ -241,7 +251,7 @@ function DatePicker({ value, onChange, placeholder, minDate }: DatePickerProps)
241
251
  );
242
252
 
243
253
  const formatDate = (date: Date) => {
244
- return date.toLocaleDateString(undefined, {
254
+ return date.toLocaleDateString(locale, {
245
255
  year: "numeric",
246
256
  month: "short",
247
257
  day: "numeric",
@@ -249,22 +259,36 @@ function DatePicker({ value, onChange, placeholder, minDate }: DatePickerProps)
249
259
  };
250
260
 
251
261
  return (
252
- <Popover open={open} onOpenChange={setOpen}>
253
- <PopoverTrigger asChild>
254
- <Button variant="outline" size="sm" className="h-8 w-[130px] cursor-pointer justify-start gap-2 font-normal">
255
- <CalendarIcon className="h-3.5 w-3.5 text-muted-foreground" />
256
- {value ? formatDate(value) : <span className="text-muted-foreground">{placeholder}</span>}
262
+ <div className="flex items-center gap-1">
263
+ <Popover open={open} onOpenChange={setOpen}>
264
+ <PopoverTrigger asChild>
265
+ <Button variant="outline" size="sm" className="h-8 w-[130px] cursor-pointer justify-start gap-2 font-normal">
266
+ <CalendarIcon className="h-3.5 w-3.5 text-muted-foreground" />
267
+ {value ? formatDate(value) : <span className="text-muted-foreground">{placeholder}</span>}
268
+ </Button>
269
+ </PopoverTrigger>
270
+ <PopoverContent className="w-auto p-0" align="start">
271
+ <Calendar
272
+ mode="single"
273
+ selected={value}
274
+ onSelect={handleSelect}
275
+ disabled={minDate ? { before: minDate } : undefined}
276
+ />
277
+ </PopoverContent>
278
+ </Popover>
279
+ {value ? (
280
+ <Button
281
+ variant="ghost"
282
+ size="sm"
283
+ className="h-8 w-8 cursor-pointer p-0"
284
+ onClick={() => onChange(undefined)}
285
+ aria-label={`${t("Clear filters")} ${placeholder}`}
286
+ title={`${t("Clear filters")} ${placeholder}`}
287
+ >
288
+ <XIcon className="h-3.5 w-3.5" />
257
289
  </Button>
258
- </PopoverTrigger>
259
- <PopoverContent className="w-auto p-0" align="start">
260
- <Calendar
261
- mode="single"
262
- selected={value}
263
- onSelect={handleSelect}
264
- disabled={minDate ? { before: minDate } : undefined}
265
- />
266
- </PopoverContent>
267
- </Popover>
290
+ ) : null}
291
+ </div>
268
292
  );
269
293
  }
270
294
 
@@ -277,8 +301,7 @@ type StatusFilterSectionProps = {
277
301
  function StatusFilterSection({ value, onChange, t }: StatusFilterSectionProps) {
278
302
  const handleChange = useCallback(
279
303
  (status: string) => {
280
- // Empty string means "clear selection"
281
- if (status === "") {
304
+ if (status === "all") {
282
305
  onChange([]);
283
306
  } else {
284
307
  onChange([status as StatusFilter]);
@@ -298,10 +321,16 @@ function StatusFilterSection({ value, onChange, t }: StatusFilterSectionProps) {
298
321
  <div className="space-y-3">
299
322
  <Label className="font-medium text-muted-foreground text-xs uppercase">{t("Status")}</Label>
300
323
  <RadioGroup
301
- value={value[0] ?? ""}
324
+ value={value[0] ?? "all"}
302
325
  onValueChange={(v) => handleChange(v as string)}
303
326
  className="flex h-8 flex-wrap items-center gap-4"
304
327
  >
328
+ <div className="flex cursor-pointer items-center gap-2">
329
+ <RadioGroupItem value="all" id="status-all" className="cursor-pointer" />
330
+ <Label htmlFor="status-all" className="cursor-pointer font-normal text-sm">
331
+ {t("All statuses")}
332
+ </Label>
333
+ </div>
305
334
  {STATUS_OPTIONS.map((status) => (
306
335
  <div key={status} className="flex cursor-pointer items-center gap-2">
307
336
  <RadioGroupItem value={status} id={`status-${status}`} className="cursor-pointer" />
@@ -113,6 +113,55 @@ export function buildQueryFromFilterState(state: FilterState | null): string | u
113
113
  return JSON.stringify(query);
114
114
  }
115
115
 
116
+ function parseQueryObject(query: string | undefined): Record<string, unknown> | null {
117
+ if (!query) return null;
118
+
119
+ try {
120
+ const parsed = JSON.parse(query);
121
+ if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) return null;
122
+ return parsed as Record<string, unknown>;
123
+ } catch {
124
+ return null;
125
+ }
126
+ }
127
+
128
+ function mergeApiQueries(baseQuery: string | undefined, filterQuery: string | undefined): string | undefined {
129
+ if (!baseQuery) return filterQuery;
130
+ if (!filterQuery) return baseQuery;
131
+
132
+ const parsedBase = parseQueryObject(baseQuery);
133
+ const parsedFilter = parseQueryObject(filterQuery);
134
+
135
+ if (!parsedBase || !parsedFilter) return filterQuery;
136
+
137
+ return JSON.stringify({
138
+ ...parsedBase,
139
+ ...parsedFilter,
140
+ });
141
+ }
142
+
143
+ function syncParamsToUrl(params: TableQueryParams) {
144
+ const searchParams = new URLSearchParams();
145
+
146
+ Object.entries(params).forEach(([key, value]) => {
147
+ if (value === undefined || value === null) {
148
+ return;
149
+ }
150
+
151
+ if (Array.isArray(value)) {
152
+ for (const item of value) {
153
+ searchParams.append(key, item);
154
+ }
155
+ return;
156
+ }
157
+
158
+ searchParams.set(key, String(value));
159
+ });
160
+
161
+ const newUrl = `${window.location.pathname}${searchParams.toString() ? `?${searchParams.toString()}` : ""}`;
162
+ window.history.pushState({}, "", newUrl);
163
+ }
164
+
116
165
  /**
117
166
  * Manages table state (search, pagination, filters) with optional URL sync
118
167
  */
@@ -164,16 +213,7 @@ export function useTableState({ initialParams = {}, onChangeParams, disableUrlSy
164
213
  changeHandler(params);
165
214
  } else {
166
215
  // Update URL directly
167
- const searchParams = new URLSearchParams();
168
-
169
- Object.entries(params).forEach(([key, value]) => {
170
- if (value !== undefined && value !== null) {
171
- searchParams.set(key, String(value));
172
- }
173
- });
174
-
175
- const newUrl = `${window.location.pathname}${searchParams.toString() ? `?${searchParams.toString()}` : ""}`;
176
- window.history.pushState({}, "", newUrl);
216
+ syncParamsToUrl(params);
177
217
  }
178
218
  }, [params, disableUrlSync]);
179
219
 
@@ -233,6 +273,18 @@ export function useTableState({ initialParams = {}, onChangeParams, disableUrlSy
233
273
  }));
234
274
  }, []);
235
275
 
276
+ /**
277
+ * Handle sort change - sorting invalidates existing cursors.
278
+ */
279
+ const handleSortChange = useCallback((orderBy: TableQueryParams["order_by"]) => {
280
+ setParams((prevParams) => ({
281
+ ...prevParams,
282
+ order_by: orderBy,
283
+ prev_cursor: undefined,
284
+ next_cursor: undefined,
285
+ }));
286
+ }, []);
287
+
236
288
  /**
237
289
  * Parse current filter state from URL params
238
290
  */
@@ -254,7 +306,7 @@ export function useTableState({ initialParams = {}, onChangeParams, disableUrlSy
254
306
  * while query JSON is added for tables that use it (like invoices)
255
307
  */
256
308
  const apiParams = useMemo(() => {
257
- const query = buildQueryFromFilterState(filterState);
309
+ const query = mergeApiQueries(params.query, buildQueryFromFilterState(filterState));
258
310
  return { ...params, query };
259
311
  }, [params, filterState]);
260
312
 
@@ -265,5 +317,6 @@ export function useTableState({ initialParams = {}, onChangeParams, disableUrlSy
265
317
  handleSearch,
266
318
  handlePageChange,
267
319
  handleFilterChange,
320
+ handleSortChange,
268
321
  };
269
322
  }