@revenexx/cover 0.1.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 (408) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +39 -0
  3. package/app/api/account.ts +8 -0
  4. package/app/api/categories.ts +3 -0
  5. package/app/api/checkout.ts +6 -0
  6. package/app/api/images.ts +4 -0
  7. package/app/api/markets.ts +3 -0
  8. package/app/api/product.ts +3 -0
  9. package/app/api/products.ts +4 -0
  10. package/app/api/search.ts +4 -0
  11. package/app/app.config.ts +29 -0
  12. package/app/assets/css/cover.css +25 -0
  13. package/app/components/account/AccountMenu.vue +30 -0
  14. package/app/components/account/AccountShell.vue +18 -0
  15. package/app/components/account/AccountSidebar.vue +90 -0
  16. package/app/components/account/address/AccountAddressCard.vue +287 -0
  17. package/app/components/account/carts/AccountCartsTable.vue +284 -0
  18. package/app/components/account/dashboard/AccountDashboard.vue +351 -0
  19. package/app/components/account/directorder/AccountDirectOrder.vue +512 -0
  20. package/app/components/account/governance/AccountApprovalLimitsTable.vue +221 -0
  21. package/app/components/account/governance/AccountCostCenterDetail.vue +276 -0
  22. package/app/components/account/governance/AccountCostCentersTable.vue +252 -0
  23. package/app/components/account/governance/AccountRequisitionDetail.vue +295 -0
  24. package/app/components/account/governance/AccountRequisitionsTable.vue +255 -0
  25. package/app/components/account/governance/AccountWorkflowDetail.vue +215 -0
  26. package/app/components/account/governance/AccountWorkflowsTable.vue +168 -0
  27. package/app/components/account/governance/GovernanceApprovalLimitModal.vue +183 -0
  28. package/app/components/account/governance/GovernanceCostCenterModal.vue +188 -0
  29. package/app/components/account/governance/GovernanceWorkflowModal.vue +191 -0
  30. package/app/components/account/orderlists/AccountOrderListDetail.vue +349 -0
  31. package/app/components/account/orderlists/AccountOrderListsTable.vue +352 -0
  32. package/app/components/account/orders/AccountOrderDetail.vue +376 -0
  33. package/app/components/account/orders/AccountOrdersTable.vue +281 -0
  34. package/app/components/account/preferences/AccountPreferences.vue +50 -0
  35. package/app/components/auth/AuthForgotPasswordPanel.vue +88 -0
  36. package/app/components/auth/AuthLoginPanel.vue +103 -0
  37. package/app/components/auth/AuthLoginTeaser.vue +24 -0
  38. package/app/components/auth/AuthPageHeader.vue +21 -0
  39. package/app/components/auth/AuthRegisterPanel.vue +431 -0
  40. package/app/components/auth/AuthRegisterTeaser.vue +24 -0
  41. package/app/components/auth/AuthResetPasswordPanel.vue +115 -0
  42. package/app/components/cart/CartButton.vue +42 -0
  43. package/app/components/cart/actions/CartCostCenterModal.vue +110 -0
  44. package/app/components/cart/actions/CartExportModal.vue +82 -0
  45. package/app/components/cart/actions/CartPositionTextsModal.vue +92 -0
  46. package/app/components/cart/actions/CartSaveToListModal.vue +177 -0
  47. package/app/components/cart/actions/CartWorkflowPickerModal.vue +64 -0
  48. package/app/components/cart/drawer/CartDrawer.vue +49 -0
  49. package/app/components/cart/item/CartItem.vue +318 -0
  50. package/app/components/cart/manage/CartNameModal.vue +68 -0
  51. package/app/components/cart/position/CartPosition.vue +369 -0
  52. package/app/components/cart/quickadd/CartQuickAdd.vue +145 -0
  53. package/app/components/cart/recommend/CartCrossSell.vue +102 -0
  54. package/app/components/cart/recommend/CartRecommendCard.vue +59 -0
  55. package/app/components/cart/recommend/CartReorderStrip.vue +108 -0
  56. package/app/components/cart/requisition/CartRequisitionGroup.vue +74 -0
  57. package/app/components/cart/skeleton/CartSkeleton.vue +18 -0
  58. package/app/components/cart/summary/CartSummary.vue +25 -0
  59. package/app/components/cart/summary/CartSummaryBreakdown.vue +35 -0
  60. package/app/components/cart/switcher/CartSwitcher.vue +147 -0
  61. package/app/components/cart/target/CartTargetModal.vue +129 -0
  62. package/app/components/cart/view/CartApprovalsPanel.vue +89 -0
  63. package/app/components/cart/view/CartListHeader.vue +86 -0
  64. package/app/components/cart/view/CartPositionList.vue +94 -0
  65. package/app/components/cart/view/CartSummaryPanel.vue +254 -0
  66. package/app/components/cart/view/CartView.vue +310 -0
  67. package/app/components/category/info/CategoryPrice.vue +47 -0
  68. package/app/components/category/info/CategorySku.vue +15 -0
  69. package/app/components/category/info/CategoryStock.vue +25 -0
  70. package/app/components/category/list/CategoryItemCount.vue +27 -0
  71. package/app/components/category/list/CategoryListPagination.vue +81 -0
  72. package/app/components/category/skeleton/CategoryDetailSkeleton.vue +12 -0
  73. package/app/components/category/skeleton/CategoryListSkeleton.vue +65 -0
  74. package/app/components/category/view/CategoryViewToggle.vue +33 -0
  75. package/app/components/category/view/GridView.vue +78 -0
  76. package/app/components/category/view/ListView.vue +80 -0
  77. package/app/components/checkout/confirmation/CheckoutConfirmationView.vue +289 -0
  78. package/app/components/checkout/form/CheckoutSchemaForm.vue +72 -0
  79. package/app/components/checkout/onepage/CheckoutAddressPickerModal.vue +260 -0
  80. package/app/components/checkout/onepage/CheckoutAddressSection.vue +186 -0
  81. package/app/components/checkout/onepage/CheckoutDeliverySection.vue +158 -0
  82. package/app/components/checkout/onepage/CheckoutGate.vue +67 -0
  83. package/app/components/checkout/onepage/CheckoutGuestDetailsSection.vue +154 -0
  84. package/app/components/checkout/onepage/CheckoutNotesSection.vue +62 -0
  85. package/app/components/checkout/onepage/CheckoutOnePageView.vue +77 -0
  86. package/app/components/checkout/onepage/CheckoutPaymentSection.vue +181 -0
  87. package/app/components/checkout/onepage/CheckoutSectionTitle.vue +18 -0
  88. package/app/components/checkout/onepage/CheckoutSummarySection.vue +211 -0
  89. package/app/components/checkout/onepage/CheckoutWorkflowPanel.vue +38 -0
  90. package/app/components/layout/AppLogo.vue +21 -0
  91. package/app/components/layout/footer/Footer.vue +148 -0
  92. package/app/components/layout/header/AppBar.vue +75 -0
  93. package/app/components/layout/header/CategoryNav.vue +93 -0
  94. package/app/components/layout/header/HeaderActionItem.vue +81 -0
  95. package/app/components/layout/header/LocaleSwitcher.vue +123 -0
  96. package/app/components/layout/header/MainNav.vue +61 -0
  97. package/app/components/layout/header/MobileMenu.vue +81 -0
  98. package/app/components/layout/header/TopBar.vue +63 -0
  99. package/app/components/listing/ListingView.vue +227 -0
  100. package/app/components/product/detail/ProductDetailAddToCart.vue +188 -0
  101. package/app/components/product/detail/ProductDetailDescription.vue +20 -0
  102. package/app/components/product/detail/ProductDetailImageGallery.vue +58 -0
  103. package/app/components/product/detail/ProductDetailPricing.vue +108 -0
  104. package/app/components/product/detail/ProductDetailSaveToList.vue +291 -0
  105. package/app/components/product/detail/ProductDetailSpecifications.vue +33 -0
  106. package/app/components/product/detail/ProductDetailStock.vue +36 -0
  107. package/app/components/search/filter/SearchFacets.vue +256 -0
  108. package/app/components/search/filter/SearchFacetsDrawer.vue +58 -0
  109. package/app/components/ui/AppShell.vue +7 -0
  110. package/app/components/ui/debug/PiniaStateCard.vue +106 -0
  111. package/app/components/ui/feedback/EmptyState.vue +30 -0
  112. package/app/components/ui/feedback/ErrorBoundary.vue +44 -0
  113. package/app/components/ui/feedback/NotFound.vue +34 -0
  114. package/app/components/ui/forms/CountrySelection.vue +14 -0
  115. package/app/components/ui/forms/FormKitDatePicker.vue +124 -0
  116. package/app/components/ui/forms/PasswordInput.vue +136 -0
  117. package/app/components/ui/forms/SearchInput.vue +213 -0
  118. package/app/components/ui/table/TableSortButton.vue +51 -0
  119. package/app/components/ui/table/TableToolbar.vue +66 -0
  120. package/app/composables/useAccount.ts +27 -0
  121. package/app/composables/useAccountAddresses.ts +116 -0
  122. package/app/composables/useAccountOrders.ts +18 -0
  123. package/app/composables/useAppUrl.ts +38 -0
  124. package/app/composables/useAuthStore.ts +202 -0
  125. package/app/composables/useB2BContext.ts +23 -0
  126. package/app/composables/useCartActions.ts +25 -0
  127. package/app/composables/useCartCalculation.ts +90 -0
  128. package/app/composables/useCartSelection.ts +46 -0
  129. package/app/composables/useCartStore.ts +837 -0
  130. package/app/composables/useCartSummaryFormatting.ts +28 -0
  131. package/app/composables/useCategories.ts +29 -0
  132. package/app/composables/useCheckoutOnePage.ts +438 -0
  133. package/app/composables/useCheckoutProfileSource.ts +8 -0
  134. package/app/composables/useCheckoutSchema.ts +28 -0
  135. package/app/composables/useDataTable.ts +14 -0
  136. package/app/composables/useFormValidation.ts +29 -0
  137. package/app/composables/useIcons.ts +17 -0
  138. package/app/composables/useLocalePreferences.ts +45 -0
  139. package/app/composables/useMarkets.ts +15 -0
  140. package/app/composables/useProduct.ts +89 -0
  141. package/app/composables/useProductCard.ts +8 -0
  142. package/app/composables/useProductCardActions.ts +106 -0
  143. package/app/composables/useProductImage.ts +12 -0
  144. package/app/composables/useProductListing.ts +210 -0
  145. package/app/composables/useProductUrl.ts +17 -0
  146. package/app/composables/useProducts.ts +130 -0
  147. package/app/composables/useSearch.ts +12 -0
  148. package/app/composables/useThemeStore.ts +81 -0
  149. package/app/composables/useViewMode.ts +18 -0
  150. package/app/config/countries.ts +74 -0
  151. package/app/config/icons.ts +283 -0
  152. package/app/config/navigation.ts +191 -0
  153. package/app/config/palette.ts +13 -0
  154. package/app/config/themes.ts +20 -0
  155. package/app/formkit.config.ts +50 -0
  156. package/app/interfaces/account/address-list.ts +34 -0
  157. package/app/interfaces/account/order-list.ts +47 -0
  158. package/app/interfaces/account/order-lists.ts +35 -0
  159. package/app/interfaces/account/profile.ts +17 -0
  160. package/app/interfaces/account.ts +3 -0
  161. package/app/interfaces/address.ts +9 -0
  162. package/app/interfaces/auth.ts +104 -0
  163. package/app/interfaces/b2b.ts +234 -0
  164. package/app/interfaces/cart-calculation.ts +131 -0
  165. package/app/interfaces/cart-item.ts +45 -0
  166. package/app/interfaces/checkout-draft.ts +4 -0
  167. package/app/interfaces/checkout.ts +43 -0
  168. package/app/interfaces/delivery.ts +13 -0
  169. package/app/interfaces/market.ts +24 -0
  170. package/app/interfaces/payment.ts +19 -0
  171. package/app/interfaces/persisted-cart.ts +10 -0
  172. package/app/interfaces/product-detail.ts +54 -0
  173. package/app/interfaces/product-list.ts +33 -0
  174. package/app/interfaces/search-facets.ts +14 -0
  175. package/app/interfaces/validation.ts +14 -0
  176. package/app/layouts/default.vue +78 -0
  177. package/app/layouts/focus.vue +80 -0
  178. package/app/plugins/formkit-locale.client.ts +23 -0
  179. package/app/shared/constants.ts +1 -0
  180. package/app/validations/companyName.ts +18 -0
  181. package/app/validations/emailFormat.ts +10 -0
  182. package/app/validations/formValidationConfig.ts +50 -0
  183. package/app/validations/maxLength.ts +12 -0
  184. package/app/validations/minLength.ts +9 -0
  185. package/app/validations/optionalCompanyName.ts +23 -0
  186. package/app/validations/passwordsMatch.ts +5 -0
  187. package/app/validations/phoneNumber.ts +19 -0
  188. package/app/validations/required.ts +11 -0
  189. package/app/validations/termsRequired.ts +4 -0
  190. package/app/validations/types.ts +10 -0
  191. package/app/validations/zipCode.ts +15 -0
  192. package/i18n/locales/de/account.json +458 -0
  193. package/i18n/locales/de/auth.json +155 -0
  194. package/i18n/locales/de/cart.json +263 -0
  195. package/i18n/locales/de/checkout.json +217 -0
  196. package/i18n/locales/de/common.json +80 -0
  197. package/i18n/locales/de/cover.json +33 -0
  198. package/i18n/locales/de/order.json +1 -0
  199. package/i18n/locales/de/product.json +71 -0
  200. package/i18n/locales/de/search.json +54 -0
  201. package/i18n/locales/de/validation.json +12 -0
  202. package/i18n/locales/en/account.json +458 -0
  203. package/i18n/locales/en/auth.json +155 -0
  204. package/i18n/locales/en/cart.json +263 -0
  205. package/i18n/locales/en/checkout.json +217 -0
  206. package/i18n/locales/en/common.json +80 -0
  207. package/i18n/locales/en/cover.json +33 -0
  208. package/i18n/locales/en/order.json +1 -0
  209. package/i18n/locales/en/product.json +71 -0
  210. package/i18n/locales/en/search.json +54 -0
  211. package/i18n/locales/en/validation.json +12 -0
  212. package/nuxt.config.ts +109 -0
  213. package/package.json +65 -0
  214. package/public/img/product-placeholder.svg +8 -0
  215. package/public/templates/direct-order-template.csv +3 -0
  216. package/public/templates/direct-order-template.xlsx +0 -0
  217. package/server/api/account/address/[id].delete.ts +51 -0
  218. package/server/api/account/address/[id].put.ts +83 -0
  219. package/server/api/account/address/index.post.ts +60 -0
  220. package/server/api/account/addresses.get.ts +25 -0
  221. package/server/api/account/context.get.ts +16 -0
  222. package/server/api/account/governance/[domain]/[id].put.ts +57 -0
  223. package/server/api/account/governance/[domain].post.ts +111 -0
  224. package/server/api/account/order-lists/[id].delete.ts +22 -0
  225. package/server/api/account/order-lists/[id].get.ts +17 -0
  226. package/server/api/account/order-lists/[id].put.ts +46 -0
  227. package/server/api/account/order-lists/index.get.ts +14 -0
  228. package/server/api/account/order-lists/index.post.ts +38 -0
  229. package/server/api/account/orders/[id].get.ts +35 -0
  230. package/server/api/account/orders.get.ts +35 -0
  231. package/server/api/account/profile.get.ts +56 -0
  232. package/server/api/account/profile.put.ts +128 -0
  233. package/server/api/account/requisitions/[id].get.ts +15 -0
  234. package/server/api/account/requisitions.get.ts +23 -0
  235. package/server/api/auth/login.post.ts +37 -0
  236. package/server/api/auth/logout.post.ts +4 -0
  237. package/server/api/auth/me.get.ts +3 -0
  238. package/server/api/auth/personas.get.ts +7 -0
  239. package/server/api/auth/recovery.post.ts +32 -0
  240. package/server/api/auth/recovery.put.ts +37 -0
  241. package/server/api/auth/register.post.ts +126 -0
  242. package/server/api/cart/calculate.post.ts +25 -0
  243. package/server/api/cart/export.post.ts +81 -0
  244. package/server/api/cart/index.delete.ts +10 -0
  245. package/server/api/cart/index.get.ts +10 -0
  246. package/server/api/cart/sync.post.ts +14 -0
  247. package/server/api/carts/[id]/activate.post.ts +18 -0
  248. package/server/api/carts/[id]/index.delete.ts +18 -0
  249. package/server/api/carts/[id]/index.put.ts +19 -0
  250. package/server/api/carts/[id]/items.post.ts +21 -0
  251. package/server/api/carts/index.get.ts +14 -0
  252. package/server/api/carts/index.post.ts +14 -0
  253. package/server/api/categories/[slug].get.ts +22 -0
  254. package/server/api/categories/index.get.ts +11 -0
  255. package/server/api/checkout/profile.get.ts +21 -0
  256. package/server/api/checkout/schema/[key].get.ts +19 -0
  257. package/server/api/checkout/session.post.ts +8 -0
  258. package/server/api/images/detail/[filename].get.ts +18 -0
  259. package/server/api/images/list/[filename].get.ts +17 -0
  260. package/server/api/markets.get.ts +9 -0
  261. package/server/api/orders/index.post.ts +376 -0
  262. package/server/api/payment/methods.post.ts +65 -0
  263. package/server/api/product/[id].get.ts +29 -0
  264. package/server/api/products/[id].get.ts +30 -0
  265. package/server/api/products/index.get.ts +21 -0
  266. package/server/api/requisitions/index.post.ts +67 -0
  267. package/server/api/shipping/rates.post.ts +70 -0
  268. package/server/api/themes/index.get.ts +9 -0
  269. package/server/api/typesense/drop.get.ts +22 -0
  270. package/server/api/typesense/health.get.ts +9 -0
  271. package/server/api/typesense/search.post.ts +199 -0
  272. package/server/api/typesense/seed.get.ts +112 -0
  273. package/server/api/typesense/suggest.post.ts +69 -0
  274. package/server/config/account/organization.json +146 -0
  275. package/server/config/account/personas.json +169 -0
  276. package/server/config/account/user.json +8 -0
  277. package/server/config/account/workflows.json +19 -0
  278. package/server/data/account/address-list.json +103 -0
  279. package/server/data/account/order-list.json +491 -0
  280. package/server/data/account/order-lists.json +149 -0
  281. package/server/data/account/profile.json +9 -0
  282. package/server/data/account/registration-requests.json +3 -0
  283. package/server/data/account/requisitions.json +686 -0
  284. package/server/data/categories.json +186 -0
  285. package/server/data/forms/checkout/add-address.json +24 -0
  286. package/server/data/list/5137-1.png +0 -0
  287. package/server/data/list/5498-1.png +0 -0
  288. package/server/data/list/5498-2.png +0 -0
  289. package/server/data/list/5498-3.png +0 -0
  290. package/server/data/list/5498-4.png +0 -0
  291. package/server/data/list/5498-5.png +0 -0
  292. package/server/data/list/5498-6.png +0 -0
  293. package/server/data/list/5519-1.png +0 -0
  294. package/server/data/list/5713-1.png +0 -0
  295. package/server/data/list/5789-1.png +0 -0
  296. package/server/data/list/5930-1.png +0 -0
  297. package/server/data/list/6127-1.png +0 -0
  298. package/server/data/list/6234-1.png +0 -0
  299. package/server/data/list/6238-1.png +0 -0
  300. package/server/data/list/6246-1.png +0 -0
  301. package/server/data/list/6270-1.png +0 -0
  302. package/server/data/list/6330-1.png +0 -0
  303. package/server/data/list/6336-1.png +0 -0
  304. package/server/data/list/6360-1.png +0 -0
  305. package/server/data/list/6363-1.png +0 -0
  306. package/server/data/list/6375-1.png +0 -0
  307. package/server/data/list/6385-1.png +0 -0
  308. package/server/data/list/6413-1.png +0 -0
  309. package/server/data/list/6418-1.png +0 -0
  310. package/server/data/list/6465-1.png +0 -0
  311. package/server/data/list/6477-1.png +0 -0
  312. package/server/data/list/6509-1.png +0 -0
  313. package/server/data/list/6545-1.png +0 -0
  314. package/server/data/list/6548-1.png +0 -0
  315. package/server/data/list/6566-1.png +0 -0
  316. package/server/data/list/6581-1.png +0 -0
  317. package/server/data/list/6609-1.png +0 -0
  318. package/server/data/list/6611-1.png +0 -0
  319. package/server/data/list/6641-1.png +0 -0
  320. package/server/data/list/6659-1.png +0 -0
  321. package/server/data/list/6662-1.png +0 -0
  322. package/server/data/list/6689-1.png +0 -0
  323. package/server/data/list/6698-1.png +0 -0
  324. package/server/data/list/6701-1.png +0 -0
  325. package/server/data/list/6752-1.png +0 -0
  326. package/server/data/list/6755-1.png +0 -0
  327. package/server/data/list/6837-1.png +0 -0
  328. package/server/data/list/6841-1.png +0 -0
  329. package/server/data/list/6844-1.png +0 -0
  330. package/server/data/list/6846-1.png +0 -0
  331. package/server/data/list/6886-1.png +0 -0
  332. package/server/data/list/6895-1.png +0 -0
  333. package/server/data/list/6897-1.png +0 -0
  334. package/server/data/list/6919-1.png +0 -0
  335. package/server/data/list/6977-1.png +0 -0
  336. package/server/data/list/6983-1.png +0 -0
  337. package/server/data/list/6984-1.png +0 -0
  338. package/server/data/list/6985-1.png +0 -0
  339. package/server/data/list/6986-1.png +0 -0
  340. package/server/data/list/6989-1.png +0 -0
  341. package/server/data/list/6995-1.png +0 -0
  342. package/server/data/list/6998-1.png +0 -0
  343. package/server/data/markets.json +24 -0
  344. package/server/data/product-detail.json +2450 -0
  345. package/server/data/product-list.json +2450 -0
  346. package/server/data/themes.json +8 -0
  347. package/server/interfaces/account.ts +20 -0
  348. package/server/interfaces/auth.ts +32 -0
  349. package/server/interfaces/b2bContext.ts +23 -0
  350. package/server/interfaces/cart.ts +46 -0
  351. package/server/interfaces/cartCalculation.ts +17 -0
  352. package/server/interfaces/category.ts +12 -0
  353. package/server/interfaces/checkoutProfile.ts +5 -0
  354. package/server/interfaces/log.ts +3 -0
  355. package/server/interfaces/market.ts +10 -0
  356. package/server/interfaces/product.ts +21 -0
  357. package/server/interfaces/schema.ts +20 -0
  358. package/server/interfaces/theme.ts +10 -0
  359. package/server/services/ApiAuthService.ts +138 -0
  360. package/server/services/ApiCartService.ts +254 -0
  361. package/server/services/ApiCategoryService.ts +25 -0
  362. package/server/services/ApiMarketService.ts +71 -0
  363. package/server/services/ApiProductService.ts +53 -0
  364. package/server/services/LocalFileCategoryService.ts +23 -0
  365. package/server/services/LocalFileCheckoutProfileService.ts +117 -0
  366. package/server/services/LocalFileProductService.ts +128 -0
  367. package/server/services/LocalFileSchemaService.ts +70 -0
  368. package/server/services/LocalFileThemeService.ts +18 -0
  369. package/server/services/LogService.ts +28 -0
  370. package/server/services/MockAccountService.ts +58 -0
  371. package/server/services/MockAuthService.ts +105 -0
  372. package/server/services/MockB2BContextService.ts +149 -0
  373. package/server/services/MockCartCalculationService.ts +395 -0
  374. package/server/services/MockMarketService.ts +18 -0
  375. package/server/services/SdkAccountService.ts +56 -0
  376. package/server/services/SdkAuthService.ts +83 -0
  377. package/server/services/SessionCartService.ts +31 -0
  378. package/server/utils/accountService.ts +30 -0
  379. package/server/utils/authCookie.ts +13 -0
  380. package/server/utils/authService.ts +43 -0
  381. package/server/utils/b2bService.ts +49 -0
  382. package/server/utils/cartService.ts +59 -0
  383. package/server/utils/categoryService.ts +19 -0
  384. package/server/utils/checkoutProfileService.ts +9 -0
  385. package/server/utils/checkoutSession.ts +38 -0
  386. package/server/utils/coverData.ts +84 -0
  387. package/server/utils/governanceStore.ts +38 -0
  388. package/server/utils/i18n.ts +76 -0
  389. package/server/utils/inventoryService.ts +14 -0
  390. package/server/utils/liveCatalog.ts +234 -0
  391. package/server/utils/liveInventories.ts +76 -0
  392. package/server/utils/liveOrders.ts +139 -0
  393. package/server/utils/livePrices.ts +93 -0
  394. package/server/utils/locale.ts +39 -0
  395. package/server/utils/logService.ts +19 -0
  396. package/server/utils/marketService.ts +24 -0
  397. package/server/utils/orderService.ts +14 -0
  398. package/server/utils/paymentService.ts +14 -0
  399. package/server/utils/priceService.ts +14 -0
  400. package/server/utils/productService.ts +28 -0
  401. package/server/utils/productsSchema.ts +30 -0
  402. package/server/utils/revenexxApi.ts +136 -0
  403. package/server/utils/schemaService.ts +25 -0
  404. package/server/utils/serviceMode.ts +70 -0
  405. package/server/utils/shippingService.ts +14 -0
  406. package/server/utils/shopSdk.ts +88 -0
  407. package/server/utils/themeService.ts +16 -0
  408. package/server/utils/typesense.ts +27 -0
@@ -0,0 +1,376 @@
1
+ <script setup lang="ts">
2
+ import type { AccountOrder, AccountOrderStatus, AccountOrderPaymentStatus } from "../../../interfaces/account/order-list";
3
+
4
+ /**
5
+ * Detail page content of one order (account pattern: detail = own page):
6
+ * head facts, fulfillment progress with carrier tracking, positions and
7
+ * the delivery/billing/payment block — everything a buyer needs to answer
8
+ * "where is my order?" without contacting support.
9
+ */
10
+ const props = defineProps<{
11
+ orderId: string;
12
+ }>();
13
+
14
+ const { icon } = useIcons();
15
+ const { t, d } = useI18n();
16
+ const { to } = useAppUrl();
17
+ const toast = useToast();
18
+ const cart = useCartStore();
19
+ const { getImage } = useProductImage();
20
+ const { formatCurrency } = useCartSummaryFormatting();
21
+
22
+ const { data: order, status, error } = useFetch<AccountOrder>(
23
+ `/api/account/orders/${props.orderId}`,
24
+ { key: `account-order:${props.orderId}` },
25
+ );
26
+
27
+ const statusColors: Record<AccountOrderStatus, "info" | "warning" | "primary" | "success" | "error"> = {
28
+ "processing": "info",
29
+ "approval-pending": "warning",
30
+ "shipped": "primary",
31
+ "delivered": "success",
32
+ "cancelled": "error",
33
+ };
34
+
35
+ const paymentColors: Record<AccountOrderPaymentStatus, "warning" | "success" | "neutral"> = {
36
+ open: "warning",
37
+ paid: "success",
38
+ refunded: "neutral",
39
+ };
40
+
41
+ const formatDate = (value: string) => {
42
+ const parsed = new Date(value);
43
+ return Number.isNaN(parsed.getTime()) ? value : d(parsed, { dateStyle: "medium" });
44
+ };
45
+
46
+ /* ---- fulfillment progress -------------------------------------------- */
47
+ const PROGRESS_STEPS = ["ordered", "processing", "shipped", "delivered"] as const;
48
+
49
+ const showProgress = computed(() =>
50
+ !!order.value?.progress?.length && order.value.status !== "cancelled");
51
+
52
+ const reachedIndex = computed(() => {
53
+ const milestones = order.value?.progress ?? [];
54
+ return milestones.reduce(
55
+ (max, milestone) => Math.max(max, PROGRESS_STEPS.indexOf(milestone.step)),
56
+ 0,
57
+ );
58
+ });
59
+
60
+ /** Fill centered on the reached step; a delivered order fills the bar. */
61
+ const progressWidth = computed(() => {
62
+ if (reachedIndex.value >= PROGRESS_STEPS.length - 1) {
63
+ return "100%";
64
+ }
65
+ return `${((reachedIndex.value * 2 + 1) / (PROGRESS_STEPS.length * 2)) * 100}%`;
66
+ });
67
+
68
+ const milestoneDate = (step: string): string | undefined => {
69
+ const milestone = order.value?.progress?.find(m => m.step === step);
70
+ return milestone?.date ? formatDate(milestone.date) : undefined;
71
+ };
72
+
73
+ /* ---- addresses (seeded as " · "-joined lines) ------------------------- */
74
+ const addressLines = (value: string): string[] => value.split(" · ");
75
+
76
+ async function copyTracking(): Promise<void> {
77
+ if (!order.value?.tracking) {
78
+ return;
79
+ }
80
+ await navigator.clipboard.writeText(order.value.tracking.trackingNumber);
81
+ toast.add({
82
+ title: t("orders.detail.copied"),
83
+ color: "success",
84
+ icon: icon("success"),
85
+ duration: 1500,
86
+ });
87
+ }
88
+
89
+ async function orderAgain(): Promise<void> {
90
+ if (!order.value) {
91
+ return;
92
+ }
93
+ let added = 0;
94
+ for (const position of order.value.positions) {
95
+ const ok = cart.addItem({
96
+ id: position.id,
97
+ name: position.name,
98
+ image: position.image ?? "",
99
+ sku: position.sku,
100
+ price: position.unitPrice,
101
+ categorySlug: position.categorySlug,
102
+ subcategorySlug: position.subcategorySlug,
103
+ });
104
+ if (ok && position.quantity > 1) {
105
+ const item = cart.items.find(i => i.id === position.id && !i.requisitionId);
106
+ if (item) {
107
+ cart.updateQuantityByKey(cart.keyOf(item), item.quantity + position.quantity - 1);
108
+ }
109
+ }
110
+ if (ok) {
111
+ added++;
112
+ }
113
+ }
114
+ toast.add({
115
+ title: t("orders.orderAgainSuccess", { count: added }),
116
+ color: "success",
117
+ icon: icon("success"),
118
+ duration: 2000,
119
+ });
120
+ await navigateTo(to("/cart"));
121
+ }
122
+ </script>
123
+
124
+ <template>
125
+ <div>
126
+ <USkeleton
127
+ v-if="status === 'pending'"
128
+ class="h-64 w-full rounded-xl"
129
+ />
130
+
131
+ <UiFeedbackErrorBoundary
132
+ v-else-if="error || !order"
133
+ :title="t('orders.notFound')"
134
+ :show-retry="false"
135
+ />
136
+
137
+ <div
138
+ v-else
139
+ class="space-y-6"
140
+ >
141
+ <!-- Head -->
142
+ <div
143
+ class="border border-(--ui-border) rounded-xl p-4 sm:p-5
144
+ flex flex-wrap items-center gap-x-8 gap-y-3"
145
+ >
146
+ <div>
147
+ <p class="text-xs text-muted">
148
+ {{ t('orders.card.date') }}
149
+ </p>
150
+ <p class="text-sm font-medium text-highlighted">
151
+ {{ formatDate(order.date) }}
152
+ </p>
153
+ </div>
154
+ <div v-if="order.orderNumber">
155
+ <p class="text-xs text-muted">
156
+ {{ t('orders.card.yourOrderNumber') }}
157
+ </p>
158
+ <p class="text-sm font-medium text-highlighted">
159
+ {{ order.orderNumber }}
160
+ </p>
161
+ </div>
162
+ <div>
163
+ <p class="text-xs text-muted">
164
+ {{ t('orders.card.total') }}
165
+ </p>
166
+ <p class="text-sm font-semibold text-highlighted tabular-nums">
167
+ {{ formatCurrency(order.total) }}
168
+ </p>
169
+ </div>
170
+ <div class="flex items-center gap-2 ms-auto">
171
+ <UBadge
172
+ :color="statusColors[order.status]"
173
+ variant="subtle"
174
+ :label="t(`orders.status.${order.status}`)"
175
+ />
176
+ <UBadge
177
+ :color="paymentColors[order.paymentStatus]"
178
+ variant="outline"
179
+ :label="t(`orders.payment.${order.paymentStatus}`)"
180
+ />
181
+ </div>
182
+ </div>
183
+
184
+ <UAlert
185
+ v-if="order.status === 'approval-pending' && order.approvers?.length"
186
+ color="warning"
187
+ variant="soft"
188
+ :icon="icon('warning')"
189
+ :title="t('orders.awaitingApproval', { names: order.approvers.join(', ') })"
190
+ />
191
+
192
+ <!-- Fulfillment progress + tracking -->
193
+ <section
194
+ v-if="showProgress"
195
+ class="border border-(--ui-border) rounded-xl overflow-hidden"
196
+ >
197
+ <h2 class="px-4 sm:px-5 py-3 text-sm font-semibold text-highlighted bg-elevated/60 border-b border-(--ui-border)">
198
+ {{ t('orders.detail.progressTitle') }}
199
+ </h2>
200
+ <div class="p-4 sm:p-5">
201
+ <div
202
+ aria-hidden="true"
203
+ class="overflow-hidden rounded-full bg-elevated h-2"
204
+ >
205
+ <div
206
+ class="h-full rounded-full bg-primary w-(--order-progress) transition-all"
207
+ :style="{ '--order-progress': progressWidth }"
208
+ />
209
+ </div>
210
+ <div class="mt-4 grid grid-cols-4 gap-2 text-xs sm:text-sm font-medium">
211
+ <div
212
+ v-for="(step, index) in PROGRESS_STEPS"
213
+ :key="step"
214
+ :class="[
215
+ index === 0 ? 'text-left' : index === PROGRESS_STEPS.length - 1 ? 'text-right' : 'text-center',
216
+ index <= reachedIndex ? 'text-primary' : 'text-muted',
217
+ ]"
218
+ >
219
+ <p>{{ t(`orders.detail.steps.${step}`) }}</p>
220
+ <p
221
+ v-if="milestoneDate(step)"
222
+ class="text-xs text-muted font-normal mt-0.5 max-sm:hidden"
223
+ >
224
+ {{ milestoneDate(step) }}
225
+ </p>
226
+ </div>
227
+ </div>
228
+
229
+ <div
230
+ v-if="order.tracking"
231
+ class="mt-5 pt-4 border-t border-(--ui-border) flex flex-wrap items-center gap-x-4 gap-y-2"
232
+ >
233
+ <span class="flex items-center gap-2 text-sm text-highlighted font-medium">
234
+ <UIcon
235
+ :name="icon('delivery')"
236
+ class="size-5 text-primary"
237
+ />
238
+ {{ t('orders.detail.trackingVia', { carrier: order.tracking.carrier }) }}
239
+ </span>
240
+ <span class="text-sm text-muted tabular-nums">
241
+ {{ order.tracking.trackingNumber }}
242
+ </span>
243
+ <UTooltip :text="t('orders.detail.copyTracking')">
244
+ <UButton
245
+ color="neutral"
246
+ variant="ghost"
247
+ size="xs"
248
+ tabindex="0"
249
+ :icon="icon('copy')"
250
+ :aria-label="t('orders.detail.copyTracking')"
251
+ @click="copyTracking"
252
+ />
253
+ </UTooltip>
254
+ </div>
255
+ <p
256
+ v-else
257
+ class="mt-5 pt-4 border-t border-(--ui-border) text-sm text-muted"
258
+ >
259
+ {{ t('orders.detail.noTracking') }}
260
+ </p>
261
+ </div>
262
+ </section>
263
+
264
+ <!-- Positions -->
265
+ <section class="border border-(--ui-border) rounded-xl overflow-hidden">
266
+ <h2 class="px-4 sm:px-5 py-3 text-sm font-semibold text-highlighted bg-elevated/60 border-b border-(--ui-border)">
267
+ {{ t('orders.positionsTitle') }}
268
+ </h2>
269
+ <ul class="divide-y divide-(--ui-border) px-4 sm:px-5">
270
+ <li
271
+ v-for="position in order.positions"
272
+ :key="position.id"
273
+ class="flex items-center gap-3 py-3"
274
+ >
275
+ <span
276
+ class="w-14 h-14 shrink-0 flex items-center justify-center rounded-lg
277
+ bg-elevated/60 ring-1 ring-(--ui-border) p-1.5"
278
+ >
279
+ <img
280
+ v-if="position.image"
281
+ :src="getImage(position.image)"
282
+ :alt="position.name"
283
+ class="max-h-full max-w-full object-contain"
284
+ >
285
+ </span>
286
+ <div class="min-w-0 flex-1">
287
+ <p class="text-sm font-medium text-highlighted truncate">
288
+ {{ position.name }}
289
+ </p>
290
+ <p class="text-xs text-muted">
291
+ {{ position.sku }} · {{ position.quantity }}× {{ formatCurrency(position.unitPrice) }}
292
+ </p>
293
+ </div>
294
+ <span class="text-sm tabular-nums text-highlighted shrink-0">
295
+ {{ formatCurrency(position.lineTotal) }}
296
+ </span>
297
+ </li>
298
+ </ul>
299
+ <div
300
+ class="flex justify-between px-4 sm:px-5 py-3 border-t border-(--ui-border)
301
+ text-sm font-semibold text-highlighted"
302
+ >
303
+ <span>{{ t('orders.card.total') }}</span>
304
+ <span class="tabular-nums">{{ formatCurrency(order.total) }}</span>
305
+ </div>
306
+ </section>
307
+
308
+ <!-- Delivery & payment -->
309
+ <section class="border border-(--ui-border) rounded-xl overflow-hidden">
310
+ <h2 class="px-4 sm:px-5 py-3 text-sm font-semibold text-highlighted bg-elevated/60 border-b border-(--ui-border)">
311
+ {{ t('orders.detail.infoTitle') }}
312
+ </h2>
313
+ <dl class="grid grid-cols-1 sm:grid-cols-2 gap-x-8 gap-y-6 p-4 sm:p-5 text-sm">
314
+ <div v-if="order.deliveryAddress">
315
+ <dt class="font-medium text-highlighted">
316
+ {{ t('orders.deliveryAddress') }}
317
+ </dt>
318
+ <dd class="mt-2 text-muted">
319
+ <span
320
+ v-for="line in addressLines(order.deliveryAddress)"
321
+ :key="line"
322
+ class="block"
323
+ >{{ line }}</span>
324
+ </dd>
325
+ </div>
326
+ <div v-if="order.billingAddress">
327
+ <dt class="font-medium text-highlighted">
328
+ {{ t('orders.detail.billingAddress') }}
329
+ </dt>
330
+ <dd class="mt-2 text-muted">
331
+ <span
332
+ v-for="line in addressLines(order.billingAddress)"
333
+ :key="line"
334
+ class="block"
335
+ >{{ line }}</span>
336
+ </dd>
337
+ </div>
338
+ <div v-if="order.paymentMethod">
339
+ <dt class="font-medium text-highlighted">
340
+ {{ t('orders.detail.paymentMethodLabel') }}
341
+ </dt>
342
+ <dd class="mt-2 text-muted flex items-center gap-2">
343
+ <UIcon
344
+ :name="icon('payment')"
345
+ class="size-4 text-dimmed"
346
+ />
347
+ {{ t(`paymentMethods.${order.paymentMethod}`) }}
348
+ </dd>
349
+ </div>
350
+ <div v-if="order.tracking">
351
+ <dt class="font-medium text-highlighted">
352
+ {{ t('orders.detail.trackingTitle') }}
353
+ </dt>
354
+ <dd class="mt-2 text-muted">
355
+ <span class="block">{{ order.tracking.carrier }}</span>
356
+ <span class="block tabular-nums">{{ order.tracking.trackingNumber }}</span>
357
+ </dd>
358
+ </div>
359
+ </dl>
360
+ </section>
361
+
362
+ <!-- Actions -->
363
+ <div class="flex justify-end">
364
+ <UButton
365
+ v-if="order.status !== 'cancelled'"
366
+ color="primary"
367
+ variant="soft"
368
+ tabindex="0"
369
+ :icon="icon('reset')"
370
+ :label="t('orders.orderAgain')"
371
+ @click="orderAgain"
372
+ />
373
+ </div>
374
+ </div>
375
+ </div>
376
+ </template>
@@ -0,0 +1,281 @@
1
+ <script setup lang="ts">
2
+ import type { TableColumn } from "@nuxt/ui";
3
+
4
+ import type { AccountOrder, AccountOrderStatus, AccountOrderPaymentStatus } from "../../../interfaces/account/order-list";
5
+
6
+ /**
7
+ * Order history overview as a data table (datatable-ux): search, status
8
+ * filter pills with counts and sortable columns; details live on their
9
+ * own page. "Order again" puts all positions back into the cart.
10
+ */
11
+ const { icon } = useIcons();
12
+ const { t, d } = useI18n();
13
+ const localePath = useLocalePath();
14
+ const { to } = useAppUrl();
15
+ const toast = useToast();
16
+ const cart = useCartStore();
17
+ const { orders, status, error } = useAccountOrders();
18
+ const { formatCurrency } = useCartSummaryFormatting();
19
+ const { tableUi } = useDataTable();
20
+
21
+ const statusColors: Record<AccountOrderStatus, "info" | "warning" | "primary" | "success" | "error"> = {
22
+ "processing": "info",
23
+ "approval-pending": "warning",
24
+ "shipped": "primary",
25
+ "delivered": "success",
26
+ "cancelled": "error",
27
+ };
28
+
29
+ const paymentColors: Record<AccountOrderPaymentStatus, "warning" | "success" | "neutral"> = {
30
+ open: "warning",
31
+ paid: "success",
32
+ refunded: "neutral",
33
+ };
34
+
35
+ /* ---- search + status filter ------------------------------------------- */
36
+ const search = ref("");
37
+ const statusFilter = ref<string>("all");
38
+
39
+ const searched = computed(() => {
40
+ const query = search.value.trim().toLowerCase();
41
+ if (!query) {
42
+ return orders.value;
43
+ }
44
+ return orders.value.filter(order =>
45
+ [order.id, order.orderNumber ?? "", ...order.positions.map(p => `${p.name} ${p.sku}`)]
46
+ .some(value => value.toLowerCase().includes(query)),
47
+ );
48
+ });
49
+
50
+ const filtered = computed(() =>
51
+ statusFilter.value === "all"
52
+ ? searched.value
53
+ : searched.value.filter(order => order.status === statusFilter.value),
54
+ );
55
+
56
+ const ORDER_STATUSES: AccountOrderStatus[] = [
57
+ "approval-pending",
58
+ "processing",
59
+ "shipped",
60
+ "delivered",
61
+ "cancelled",
62
+ ];
63
+
64
+ const filterItems = computed(() => [
65
+ { value: "all", label: t("table.filterAll"), count: searched.value.length },
66
+ ...ORDER_STATUSES
67
+ .map(value => ({
68
+ value,
69
+ label: t(`orders.status.${value}`),
70
+ count: searched.value.filter(order => order.status === value).length,
71
+ }))
72
+ .filter(item => item.count > 0),
73
+ ]);
74
+
75
+ /* ---- columns + sorting -------------------------------------------------- */
76
+ const sorting = ref([{ id: "date", desc: true }]);
77
+
78
+ const columns = computed<TableColumn<AccountOrder>[]>(() => [
79
+ { accessorKey: "id", header: t("orders.columns.id") },
80
+ { accessorKey: "date", header: t("orders.columns.date") },
81
+ {
82
+ id: "positions",
83
+ accessorFn: row => row.positions.length,
84
+ header: t("orders.columns.positions"),
85
+ meta: { class: { th: "text-right", td: "text-right" } },
86
+ },
87
+ {
88
+ accessorKey: "total",
89
+ header: t("orders.columns.total"),
90
+ meta: { class: { th: "text-right", td: "text-right" } },
91
+ },
92
+ { accessorKey: "status", header: t("orders.columns.status"), enableSorting: false },
93
+ { id: "payment", header: t("orders.columns.payment"), enableSorting: false },
94
+ { id: "actions", header: "", enableSorting: false },
95
+ ]);
96
+
97
+ const formatDate = (value: string) => {
98
+ const parsed = new Date(value);
99
+ return Number.isNaN(parsed.getTime()) ? value : d(parsed, { dateStyle: "medium" });
100
+ };
101
+
102
+ /* ---- order again --------------------------------------------------------- */
103
+ async function orderAgain(order: AccountOrder): Promise<void> {
104
+ let added = 0;
105
+ for (const position of order.positions) {
106
+ const ok = cart.addItem({
107
+ id: position.id,
108
+ name: position.name,
109
+ image: position.image ?? "",
110
+ sku: position.sku,
111
+ price: position.unitPrice,
112
+ categorySlug: position.categorySlug,
113
+ subcategorySlug: position.subcategorySlug,
114
+ });
115
+ if (ok && position.quantity > 1) {
116
+ const item = cart.items.find(i => i.id === position.id && !i.requisitionId);
117
+ if (item) {
118
+ cart.updateQuantityByKey(cart.keyOf(item), item.quantity + position.quantity - 1);
119
+ }
120
+ }
121
+ if (ok) {
122
+ added++;
123
+ }
124
+ }
125
+ toast.add({
126
+ title: t("orders.orderAgainSuccess", { count: added }),
127
+ color: "success",
128
+ icon: icon("success"),
129
+ duration: 2000,
130
+ });
131
+ await navigateTo(to("/cart"));
132
+ }
133
+ </script>
134
+
135
+ <template>
136
+ <div>
137
+ <UiFeedbackErrorBoundary
138
+ v-if="error"
139
+ :title="t('orders.errorTitle')"
140
+ :description="t('orders.errorDescription')"
141
+ :show-retry="false"
142
+ />
143
+
144
+ <template v-else>
145
+ <UiTableToolbar
146
+ v-model:search="search"
147
+ v-model:filter="statusFilter"
148
+ :search-placeholder="t('orders.searchPlaceholder')"
149
+ :filter-items="filterItems"
150
+ />
151
+
152
+ <UTable
153
+ v-model:sorting="sorting"
154
+ :data="filtered"
155
+ :columns="columns"
156
+ :loading="status === 'pending'"
157
+ :ui="tableUi"
158
+ >
159
+ <template #id-header="{ column }">
160
+ <UiTableSortButton
161
+ :column="column"
162
+ :label="t('orders.columns.id')"
163
+ />
164
+ </template>
165
+ <template #date-header="{ column }">
166
+ <UiTableSortButton
167
+ :column="column"
168
+ :label="t('orders.columns.date')"
169
+ />
170
+ </template>
171
+ <template #positions-header="{ column }">
172
+ <UiTableSortButton
173
+ :column="column"
174
+ :label="t('orders.columns.positions')"
175
+ />
176
+ </template>
177
+ <template #total-header="{ column }">
178
+ <UiTableSortButton
179
+ :column="column"
180
+ :label="t('orders.columns.total')"
181
+ />
182
+ </template>
183
+
184
+ <template #id-cell="{ row }">
185
+ <div class="min-w-0">
186
+ <ULink
187
+ class="font-semibold text-highlighted hover:text-primary transition-colors"
188
+ :to="localePath(`/account/orders/${row.original.id}`)"
189
+ >
190
+ {{ row.original.id }}
191
+ </ULink>
192
+ <p
193
+ v-if="row.original.orderNumber"
194
+ class="text-xs text-muted truncate"
195
+ >
196
+ {{ row.original.orderNumber }}
197
+ </p>
198
+ </div>
199
+ </template>
200
+ <template #date-cell="{ row }">
201
+ {{ formatDate(row.original.date) }}
202
+ </template>
203
+ <template #positions-cell="{ row }">
204
+ <span class="tabular-nums">{{ row.original.positions.length }}</span>
205
+ </template>
206
+ <template #total-cell="{ row }">
207
+ <span class="tabular-nums text-highlighted">{{ formatCurrency(row.original.total) }}</span>
208
+ </template>
209
+ <template #status-cell="{ row }">
210
+ <UBadge
211
+ :color="statusColors[row.original.status]"
212
+ variant="subtle"
213
+ size="sm"
214
+ :label="t(`orders.status.${row.original.status}`)"
215
+ />
216
+ </template>
217
+ <template #payment-cell="{ row }">
218
+ <UBadge
219
+ :color="paymentColors[row.original.paymentStatus]"
220
+ variant="outline"
221
+ size="sm"
222
+ :label="t(`orders.payment.${row.original.paymentStatus}`)"
223
+ />
224
+ </template>
225
+ <template #actions-cell="{ row }">
226
+ <div class="flex items-center justify-end gap-1">
227
+ <UTooltip :text="t('orders.details')">
228
+ <UButton
229
+ color="neutral"
230
+ variant="ghost"
231
+ size="sm"
232
+ tabindex="0"
233
+ :icon="icon('eye')"
234
+ :aria-label="t('orders.details')"
235
+ :to="localePath(`/account/orders/${row.original.id}`)"
236
+ />
237
+ </UTooltip>
238
+ <UTooltip
239
+ v-if="row.original.status !== 'cancelled'"
240
+ :text="t('orders.orderAgain')"
241
+ >
242
+ <UButton
243
+ color="primary"
244
+ variant="soft"
245
+ size="sm"
246
+ tabindex="0"
247
+ :icon="icon('reset')"
248
+ :aria-label="t('orders.orderAgain')"
249
+ @click="orderAgain(row.original)"
250
+ />
251
+ </UTooltip>
252
+ </div>
253
+ </template>
254
+
255
+ <template #loading>
256
+ <div class="space-y-3 py-2">
257
+ <USkeleton
258
+ v-for="index in 5"
259
+ :key="index"
260
+ class="h-9 w-full"
261
+ />
262
+ </div>
263
+ </template>
264
+ <template #empty>
265
+ <UiFeedbackEmptyState
266
+ v-if="search || statusFilter !== 'all'"
267
+ :icon="icon('no-results')"
268
+ :title="t('table.noResults.title')"
269
+ :description="t('table.noResults.description')"
270
+ />
271
+ <UiFeedbackEmptyState
272
+ v-else
273
+ :icon="icon('orders-empty')"
274
+ :title="t('orders.empty.title')"
275
+ :description="t('orders.empty.description')"
276
+ />
277
+ </template>
278
+ </UTable>
279
+ </template>
280
+ </div>
281
+ </template>
@@ -0,0 +1,50 @@
1
+ <script setup lang="ts">
2
+ /**
3
+ * Personal ordering preferences. Currently: the target-cart behavior of
4
+ * the multiple-carts feature — mirror of the "always use the active
5
+ * cart" option offered by the add-to-cart dialog.
6
+ */
7
+ const { icon } = useIcons();
8
+ const { t } = useI18n();
9
+ const toast = useToast();
10
+ const cart = useCartStore();
11
+ const { settings } = useB2BContext();
12
+
13
+ const multiCartEnabled = computed(() => settings.value?.cart.multipleCarts === true);
14
+
15
+ function onToggle(value: boolean): void {
16
+ cart.setAlwaysActiveCart(value);
17
+ toast.add({
18
+ title: t("preferences.saved"),
19
+ color: "success",
20
+ icon: icon("success"),
21
+ duration: 1500,
22
+ });
23
+ }
24
+ </script>
25
+
26
+ <template>
27
+ <div class="space-y-6">
28
+ <section class="border border-(--ui-border) rounded-xl overflow-hidden">
29
+ <h2 class="px-4 sm:px-5 py-3 text-sm font-semibold text-highlighted bg-elevated/60 border-b border-(--ui-border)">
30
+ {{ t('preferences.cartSection') }}
31
+ </h2>
32
+ <div class="p-4 sm:p-5 space-y-4">
33
+ <UAlert
34
+ v-if="!multiCartEnabled"
35
+ color="neutral"
36
+ variant="subtle"
37
+ :icon="icon('info')"
38
+ :title="t('cart.multi.page.disabled')"
39
+ />
40
+ <USwitch
41
+ :model-value="cart.alwaysActiveCart"
42
+ :disabled="!multiCartEnabled"
43
+ :label="t('preferences.alwaysActive')"
44
+ :description="t('preferences.alwaysActiveHint')"
45
+ @update:model-value="onToggle"
46
+ />
47
+ </div>
48
+ </section>
49
+ </div>
50
+ </template>