@customafk/lunas-ui 0.1.17 → 0.1.19
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.
- package/dist/{avatar-Cp9PebV0.mjs → avatar-BaxX-xI9.mjs} +1 -1
- package/dist/{avatar-Cp9PebV0.mjs.map → avatar-BaxX-xI9.mjs.map} +1 -1
- package/dist/{avatar-DzZQenXU.cjs → avatar-dAu1c0bC.cjs} +1 -1
- package/dist/{avatar-DzZQenXU.cjs.map → avatar-dAu1c0bC.cjs.map} +1 -1
- package/dist/{button-CqxqZbcQ.mjs → button-DIEXNBpi.mjs} +2 -2
- package/dist/{button-CqxqZbcQ.mjs.map → button-DIEXNBpi.mjs.map} +1 -1
- package/dist/{button-Q8sFW-l8.d.mts → button-DUf6JdVx.d.cts} +7 -7
- package/dist/{button-BLm2W-rK.d.cts → button-Dv8PU7Hv.d.mts} +7 -7
- package/dist/{button.variants-CXNnQvXs.mjs → button.variants-DmUzMbB8.mjs} +1 -1
- package/dist/{button.variants-CXNnQvXs.mjs.map → button.variants-DmUzMbB8.mjs.map} +1 -1
- package/dist/{card-BTswGEgi.mjs → card-jwcpmq7e.mjs} +1 -1
- package/dist/{card-BTswGEgi.mjs.map → card-jwcpmq7e.mjs.map} +1 -1
- package/dist/cards/grid-product-card.mjs +1 -1
- package/dist/cards/product-card.mjs +1 -1
- package/dist/cards/simple-card.mjs +1 -1
- package/dist/{close-Bb5Pi28f.mjs → close-CGei_VQ6.mjs} +1 -1
- package/dist/{close-Bb5Pi28f.mjs.map → close-CGei_VQ6.mjs.map} +1 -1
- package/dist/{close-DKsjFTou.cjs → close-CXhpIVAZ.cjs} +1 -1
- package/dist/{close-DKsjFTou.cjs.map → close-CXhpIVAZ.cjs.map} +1 -1
- package/dist/{command-Dohhg_fY.cjs → command-B8Tbe4oe.cjs} +2 -2
- package/dist/{command-Dohhg_fY.cjs.map → command-B8Tbe4oe.cjs.map} +1 -1
- package/dist/{command-D_f6JoHE.mjs → command-BLlJzBw9.mjs} +2 -2
- package/dist/{command-D_f6JoHE.mjs.map → command-BLlJzBw9.mjs.map} +1 -1
- package/dist/data-display/country.cjs +1 -1
- package/dist/data-display/country.cjs.map +1 -1
- package/dist/data-display/country.d.cts +1 -1
- package/dist/data-display/country.d.mts +1 -1
- package/dist/data-display/country.mjs +1 -1
- package/dist/data-display/country.mjs.map +1 -1
- package/dist/data-display/data-list.cjs +1 -1
- package/dist/data-display/data-list.cjs.map +1 -1
- package/dist/data-display/data-list.mjs +1 -1
- package/dist/data-display/data-list.mjs.map +1 -1
- package/dist/data-display/date-tooltip.cjs +1 -1
- package/dist/data-display/date-tooltip.mjs +1 -1
- package/dist/data-display/date.cjs +1 -1
- package/dist/data-display/date.mjs +1 -1
- package/dist/data-display/empty.cjs +1 -1
- package/dist/data-display/empty.cjs.map +1 -1
- package/dist/data-display/empty.mjs +1 -1
- package/dist/data-display/empty.mjs.map +1 -1
- package/dist/data-display/name.cjs +1 -1
- package/dist/data-display/name.cjs.map +1 -1
- package/dist/data-display/name.mjs +1 -1
- package/dist/data-display/name.mjs.map +1 -1
- package/dist/data-display/phone-number.cjs +1 -1
- package/dist/data-display/phone-number.mjs +1 -1
- package/dist/data-display/role-badge.d.cts +1 -1
- package/dist/data-display/role-badge.d.mts +1 -1
- package/dist/data-display/role-badge.mjs +1 -1
- package/dist/data-display/statistic.cjs +1 -1
- package/dist/data-display/statistic.cjs.map +1 -1
- package/dist/data-display/statistic.d.cts +2 -2
- package/dist/data-display/statistic.d.mts +2 -2
- package/dist/data-display/statistic.mjs +1 -1
- package/dist/data-display/statistic.mjs.map +1 -1
- package/dist/data-display/user.cjs +1 -1
- package/dist/data-display/user.cjs.map +1 -1
- package/dist/data-display/user.mjs +1 -1
- package/dist/data-display/user.mjs.map +1 -1
- package/dist/{date-BUKVV2NZ.cjs → date-DU8xM0hQ.cjs} +2 -2
- package/dist/date-DU8xM0hQ.cjs.map +1 -0
- package/dist/{date-DtgYKyyk.mjs → date-Vl-tZLuD.mjs} +2 -2
- package/dist/date-Vl-tZLuD.mjs.map +1 -0
- package/dist/{dialog-D5WkhH7J.d.mts → dialog-CstR_oRr.d.mts} +12 -12
- package/dist/{dialog-D4LocfVZ.d.cts → dialog-DJ_WrGvX.d.cts} +12 -12
- package/dist/{dialog-DEnVUD9A.cjs → dialog-DZ_gg6Ol.cjs} +2 -2
- package/dist/{dialog-DEnVUD9A.cjs.map → dialog-DZ_gg6Ol.cjs.map} +1 -1
- package/dist/{dialog-D5bBgIs8.mjs → dialog-DeZVOOmh.mjs} +2 -2
- package/dist/{dialog-D5bBgIs8.mjs.map → dialog-DeZVOOmh.mjs.map} +1 -1
- package/dist/dialogs/confirm-dialog.mjs +1 -1
- package/dist/dialogs/detail-dialog/component/sidebar.cjs +1 -1
- package/dist/dialogs/detail-dialog/component/sidebar.d.cts +28 -28
- package/dist/dialogs/detail-dialog/component/sidebar.d.mts +28 -28
- package/dist/dialogs/detail-dialog/component/sidebar.mjs +1 -1
- package/dist/dialogs/detail-dialog/index.cjs +1 -1
- package/dist/dialogs/detail-dialog/index.cjs.map +1 -1
- package/dist/dialogs/detail-dialog/index.mjs +1 -1
- package/dist/dialogs/detail-dialog/index.mjs.map +1 -1
- package/dist/dialogs/error-dialog.mjs +1 -1
- package/dist/dialogs/loading-dialog.cjs +1 -1
- package/dist/dialogs/loading-dialog.mjs +1 -1
- package/dist/{dist-1xnmIjvR.mjs → dist-CT2ZB1PF.mjs} +1 -1
- package/dist/{dist-1xnmIjvR.mjs.map → dist-CT2ZB1PF.mjs.map} +1 -1
- package/dist/{drawer-D8hxCiVo.mjs → drawer-272Fecsq.mjs} +1 -1
- package/dist/{drawer-D8hxCiVo.mjs.map → drawer-272Fecsq.mjs.map} +1 -1
- package/dist/{drawer-DpT2kOKz.cjs → drawer-y-Yrqskp.cjs} +1 -1
- package/dist/{drawer-DpT2kOKz.cjs.map → drawer-y-Yrqskp.cjs.map} +1 -1
- package/dist/{dropdown-menu-BuyuU6uF.cjs → dropdown-menu-C_dw8mro.cjs} +1 -1
- package/dist/{dropdown-menu-BuyuU6uF.cjs.map → dropdown-menu-C_dw8mro.cjs.map} +1 -1
- package/dist/{dropdown-menu-Bt3WFMXE.mjs → dropdown-menu-CtU6oBI4.mjs} +1 -1
- package/dist/{dropdown-menu-Bt3WFMXE.mjs.map → dropdown-menu-CtU6oBI4.mjs.map} +1 -1
- package/dist/features/search-modal/index.cjs +1 -1
- package/dist/features/search-modal/index.d.cts +2 -2
- package/dist/features/search-modal/index.d.mts +2 -2
- package/dist/features/search-modal/index.mjs +1 -1
- package/dist/features/tables/index.cjs +2 -0
- package/dist/features/tables/index.cjs.map +1 -0
- package/dist/features/tables/index.d.cts +75 -0
- package/dist/features/tables/index.d.mts +75 -0
- package/dist/features/tables/index.mjs +2 -0
- package/dist/features/tables/index.mjs.map +1 -0
- package/dist/features/tanstack-form/index.cjs +1 -1
- package/dist/features/tanstack-form/index.mjs +1 -1
- package/dist/flex-BGQqaHKh.mjs +2 -0
- package/dist/flex-BGQqaHKh.mjs.map +1 -0
- package/dist/flex-BYj-xYaw.cjs +2 -0
- package/dist/flex-BYj-xYaw.cjs.map +1 -0
- package/dist/{image-iVki2lFP.mjs → image-BwbAQRxx.mjs} +2 -2
- package/dist/{image-iVki2lFP.mjs.map → image-BwbAQRxx.mjs.map} +1 -1
- package/dist/{input-DiGjXQJb.d.cts → input-Bzo2Bgsz.d.cts} +6 -6
- package/dist/{input-D-Wrz2-9.mjs → input-CC10BdKh.mjs} +1 -1
- package/dist/{input-D-Wrz2-9.mjs.map → input-CC10BdKh.mjs.map} +1 -1
- package/dist/{input-CUGmNdG9.d.mts → input-DJh6tJbB.d.mts} +6 -6
- package/dist/{input-CvMqFdEE.cjs → input-DhUevNkc.cjs} +1 -1
- package/dist/{input-CvMqFdEE.cjs.map → input-DhUevNkc.cjs.map} +1 -1
- package/dist/{label-D6vHFlHX.cjs → label-CaBktIhh.cjs} +1 -1
- package/dist/{label-D6vHFlHX.cjs.map → label-CaBktIhh.cjs.map} +1 -1
- package/dist/{label-D7-mzz0l.mjs → label-DVfxfyG5.mjs} +1 -1
- package/dist/{label-D7-mzz0l.mjs.map → label-DVfxfyG5.mjs.map} +1 -1
- package/dist/layouts/app-layout/index.cjs +1 -1
- package/dist/layouts/app-layout/index.d.cts +28 -28
- package/dist/layouts/app-layout/index.d.mts +28 -28
- package/dist/layouts/app-layout/index.mjs +1 -1
- package/dist/layouts/cms-layout/index.cjs +2 -0
- package/dist/layouts/cms-layout/index.cjs.map +1 -0
- package/dist/layouts/cms-layout/index.d.cts +20 -0
- package/dist/layouts/cms-layout/index.d.mts +20 -0
- package/dist/layouts/cms-layout/index.mjs +2 -0
- package/dist/layouts/cms-layout/index.mjs.map +1 -0
- package/dist/layouts/flex.cjs +1 -2
- package/dist/layouts/flex.d.cts +4 -4
- package/dist/layouts/flex.d.mts +4 -4
- package/dist/layouts/flex.mjs +1 -2
- package/dist/layouts/service-layout/index.cjs +1 -1
- package/dist/layouts/service-layout/index.d.cts +13 -13
- package/dist/layouts/service-layout/index.d.mts +15 -15
- package/dist/layouts/service-layout/index.mjs +1 -1
- package/dist/layouts/service-layout/index.mjs.map +1 -1
- package/dist/pages/FeatureDeveloping.d.cts +2 -2
- package/dist/pages/FeatureDeveloping.d.mts +2 -2
- package/dist/pages/FeatureDeveloping.mjs +1 -1
- package/dist/pages/FeatureFixing.d.cts +2 -2
- package/dist/pages/FeatureFixing.d.mts +2 -2
- package/dist/pages/FeatureFixing.mjs +1 -1
- package/dist/pages/NotAuthorized.d.cts +2 -2
- package/dist/pages/NotAuthorized.d.mts +2 -2
- package/dist/pages/NotAuthorized.mjs +1 -1
- package/dist/pages/NotFound.d.cts +2 -2
- package/dist/pages/NotFound.d.mts +2 -2
- package/dist/pages/NotFound.mjs +1 -1
- package/dist/paragraph-0_oMnSMM.cjs +2 -0
- package/dist/paragraph-0_oMnSMM.cjs.map +1 -0
- package/dist/paragraph-BXY43iOX.mjs +2 -0
- package/dist/paragraph-BXY43iOX.mjs.map +1 -0
- package/dist/resizable-B9HHrfKn.mjs +2 -0
- package/dist/resizable-B9HHrfKn.mjs.map +1 -0
- package/dist/resizable-D8XbQdj_.cjs +2 -0
- package/dist/resizable-D8XbQdj_.cjs.map +1 -0
- package/dist/{scroll-area-CROqtJiH.mjs → scroll-area-CiMS2UY7.mjs} +1 -1
- package/dist/{scroll-area-CROqtJiH.mjs.map → scroll-area-CiMS2UY7.mjs.map} +1 -1
- package/dist/{scroll-area-BzXnYZNO.cjs → scroll-area-qE6GqGkn.cjs} +1 -1
- package/dist/{scroll-area-BzXnYZNO.cjs.map → scroll-area-qE6GqGkn.cjs.map} +1 -1
- package/dist/{separator-CZM2q-Ac.mjs → separator-C2yOD6r_.mjs} +1 -1
- package/dist/{separator-CZM2q-Ac.mjs.map → separator-C2yOD6r_.mjs.map} +1 -1
- package/dist/{separator-D1uPzKkT.d.mts → separator-CY141ab_.d.cts} +3 -3
- package/dist/{separator-WeqUQUgF.d.cts → separator-CbnkvGgQ.d.mts} +3 -3
- package/dist/{separator-jytY1CPq.cjs → separator-D3CGKwyf.cjs} +1 -1
- package/dist/{separator-jytY1CPq.cjs.map → separator-D3CGKwyf.cjs.map} +1 -1
- package/dist/{sheet-CUMpo5F_.mjs → sheet-DXLayspn.mjs} +1 -1
- package/dist/{sheet-CUMpo5F_.mjs.map → sheet-DXLayspn.mjs.map} +1 -1
- package/dist/{sheet-BTGsz390.cjs → sheet-DtWL6WcJ.cjs} +1 -1
- package/dist/{sheet-BTGsz390.cjs.map → sheet-DtWL6WcJ.cjs.map} +1 -1
- package/dist/{skeleton-C02GuVQz.mjs → skeleton-CvZncisJ.mjs} +1 -1
- package/dist/{skeleton-C02GuVQz.mjs.map → skeleton-CvZncisJ.mjs.map} +1 -1
- package/dist/{tabs-BFHJXVlX.mjs → tabs-Cjh2jtUZ.mjs} +1 -1
- package/dist/{tabs-BFHJXVlX.mjs.map → tabs-Cjh2jtUZ.mjs.map} +1 -1
- package/dist/{tabs-bXy8hbAR.cjs → tabs-DSofRReg.cjs} +1 -1
- package/dist/{tabs-bXy8hbAR.cjs.map → tabs-DSofRReg.cjs.map} +1 -1
- package/dist/{title-BaHImoIu.cjs → title-C4XNLXSe.cjs} +1 -1
- package/dist/{title-BaHImoIu.cjs.map → title-C4XNLXSe.cjs.map} +1 -1
- package/dist/{title-B6SztKEv.mjs → title-CMJS7QS7.mjs} +1 -1
- package/dist/{title-B6SztKEv.mjs.map → title-CMJS7QS7.mjs.map} +1 -1
- package/dist/{tooltip-B7cSlbI6.mjs → tooltip-D0XDvxqi.mjs} +1 -1
- package/dist/{tooltip-B7cSlbI6.mjs.map → tooltip-D0XDvxqi.mjs.map} +1 -1
- package/dist/{tooltip-BCmg3WjF.d.mts → tooltip-DbdlL_22.d.mts} +6 -6
- package/dist/{tooltip-CT8tMc9R.d.cts → tooltip-JB8ZVcSM.d.cts} +6 -6
- package/dist/{tooltip-CUSQdxdt.cjs → tooltip-zYWazsY5.cjs} +1 -1
- package/dist/{tooltip-CUSQdxdt.cjs.map → tooltip-zYWazsY5.cjs.map} +1 -1
- package/dist/{types-BpxD38pa.mjs → types-BZefxrvC.mjs} +1 -1
- package/dist/{types-BpxD38pa.mjs.map → types-BZefxrvC.mjs.map} +1 -1
- package/dist/{types-CIlhkWg6.d.mts → types-D2kMOc_b.d.mts} +1 -1
- package/dist/{types-DfDk2h0N.d.cts → types-D8mZhixM.d.cts} +1 -1
- package/dist/typography/paragraph.cjs +1 -2
- package/dist/typography/paragraph.d.cts +2 -2
- package/dist/typography/paragraph.d.mts +2 -2
- package/dist/typography/paragraph.mjs +1 -2
- package/dist/typography/title.cjs +1 -1
- package/dist/typography/title.d.cts +2 -2
- package/dist/typography/title.d.mts +2 -2
- package/dist/typography/title.mjs +1 -1
- package/dist/ui/alert-dialog.d.cts +12 -12
- package/dist/ui/alert-dialog.d.mts +12 -12
- package/dist/ui/alert-dialog.mjs +1 -1
- package/dist/ui/alert.d.cts +7 -7
- package/dist/ui/alert.d.mts +7 -7
- package/dist/ui/aspect-ratio.d.cts +2 -2
- package/dist/ui/aspect-ratio.d.mts +2 -2
- package/dist/ui/avatar.cjs +1 -1
- package/dist/ui/avatar.d.cts +4 -4
- package/dist/ui/avatar.d.mts +4 -4
- package/dist/ui/avatar.mjs +1 -1
- package/dist/ui/badge.d.cts +4 -4
- package/dist/ui/badge.d.mts +4 -4
- package/dist/ui/breadcrumb.d.cts +8 -8
- package/dist/ui/breadcrumb.d.mts +8 -8
- package/dist/ui/breadcrumb.mjs +1 -1
- package/dist/ui/button-group.cjs +1 -1
- package/dist/ui/button-group.d.cts +7 -7
- package/dist/ui/button-group.d.mts +7 -7
- package/dist/ui/button-group.mjs +1 -1
- package/dist/ui/button.d.cts +1 -1
- package/dist/ui/button.d.mts +1 -1
- package/dist/ui/button.mjs +1 -1
- package/dist/ui/buttons/add-new.cjs +1 -1
- package/dist/ui/buttons/add-new.cjs.map +1 -1
- package/dist/ui/buttons/add-new.mjs +1 -1
- package/dist/ui/buttons/add-new.mjs.map +1 -1
- package/dist/ui/buttons/edit.mjs +1 -1
- package/dist/ui/buttons/refresh.cjs +1 -1
- package/dist/ui/buttons/refresh.cjs.map +1 -1
- package/dist/ui/buttons/refresh.mjs +1 -1
- package/dist/ui/buttons/refresh.mjs.map +1 -1
- package/dist/ui/buttons/trash.mjs +1 -1
- package/dist/ui/buttons/upload-image.mjs +1 -1
- package/dist/ui/calendar.d.cts +4 -4
- package/dist/ui/calendar.d.mts +4 -4
- package/dist/ui/calendar.mjs +1 -1
- package/dist/ui/card.d.cts +8 -8
- package/dist/ui/card.d.mts +8 -8
- package/dist/ui/card.mjs +1 -1
- package/dist/ui/carousel.d.cts +7 -7
- package/dist/ui/carousel.d.mts +7 -7
- package/dist/ui/carousel.mjs +1 -1
- package/dist/ui/collapsible.d.cts +4 -4
- package/dist/ui/collapsible.d.mts +4 -4
- package/dist/ui/command.cjs +1 -1
- package/dist/ui/command.d.cts +11 -11
- package/dist/ui/command.d.mts +11 -11
- package/dist/ui/command.mjs +1 -1
- package/dist/ui/context-menu.d.cts +16 -16
- package/dist/ui/context-menu.d.mts +16 -16
- package/dist/ui/dialog.cjs +1 -1
- package/dist/ui/dialog.d.cts +1 -1
- package/dist/ui/dialog.d.mts +1 -1
- package/dist/ui/dialog.mjs +1 -1
- package/dist/ui/drawer.cjs +1 -1
- package/dist/ui/drawer.d.cts +11 -11
- package/dist/ui/drawer.d.mts +11 -11
- package/dist/ui/drawer.mjs +1 -1
- package/dist/ui/dropdown-menu.cjs +1 -1
- package/dist/ui/dropdown-menu.d.cts +16 -16
- package/dist/ui/dropdown-menu.d.mts +16 -16
- package/dist/ui/dropdown-menu.mjs +1 -1
- package/dist/ui/empty.d.cts +9 -9
- package/dist/ui/empty.d.mts +9 -9
- package/dist/ui/field.cjs +1 -1
- package/dist/ui/field.d.cts +24 -24
- package/dist/ui/field.d.mts +13 -13
- package/dist/ui/field.mjs +1 -1
- package/dist/ui/file-uploader.d.cts +2 -2
- package/dist/ui/file-uploader.d.mts +2 -2
- package/dist/ui/file-uploader.mjs +1 -1
- package/dist/ui/form.cjs +1 -1
- package/dist/ui/form.d.cts +7 -7
- package/dist/ui/form.d.mts +7 -7
- package/dist/ui/form.mjs +1 -1
- package/dist/ui/hover-card.d.cts +4 -4
- package/dist/ui/hover-card.d.mts +4 -4
- package/dist/ui/image.mjs +1 -1
- package/dist/ui/input-otp.d.cts +5 -5
- package/dist/ui/input-otp.d.mts +5 -5
- package/dist/ui/input.cjs +1 -1
- package/dist/ui/input.d.cts +1 -1
- package/dist/ui/input.d.mts +1 -1
- package/dist/ui/input.mjs +1 -1
- package/dist/ui/inputs/search-input.cjs +1 -1
- package/dist/ui/inputs/search-input.d.cts +3 -3
- package/dist/ui/inputs/search-input.d.mts +3 -3
- package/dist/ui/inputs/search-input.mjs +1 -1
- package/dist/ui/item.cjs +1 -1
- package/dist/ui/item.d.cts +17 -17
- package/dist/ui/item.d.mts +14 -14
- package/dist/ui/item.mjs +1 -1
- package/dist/ui/label.cjs +1 -1
- package/dist/ui/label.d.cts +2 -2
- package/dist/ui/label.d.mts +2 -2
- package/dist/ui/label.mjs +1 -1
- package/dist/ui/menubar.d.cts +17 -17
- package/dist/ui/menubar.d.mts +17 -17
- package/dist/ui/multi-select.cjs +1 -1
- package/dist/ui/multi-select.d.mts +3 -3
- package/dist/ui/multi-select.mjs +1 -1
- package/dist/ui/navigation-menu.d.cts +11 -11
- package/dist/ui/navigation-menu.d.mts +11 -11
- package/dist/ui/pagination.d.cts +9 -9
- package/dist/ui/pagination.d.mts +9 -9
- package/dist/ui/pagination.mjs +1 -1
- package/dist/ui/popover.d.cts +5 -5
- package/dist/ui/popover.d.mts +5 -5
- package/dist/ui/progress.d.cts +2 -2
- package/dist/ui/progress.d.mts +2 -2
- package/dist/ui/radio-group.d.cts +3 -3
- package/dist/ui/radio-group.d.mts +3 -3
- package/dist/ui/resizable.cjs +1 -2
- package/dist/ui/resizable.d.cts +4 -4
- package/dist/ui/resizable.d.mts +4 -4
- package/dist/ui/resizable.mjs +1 -2
- package/dist/ui/scroll-area.cjs +1 -1
- package/dist/ui/scroll-area.d.cts +6 -6
- package/dist/ui/scroll-area.d.mts +6 -6
- package/dist/ui/scroll-area.mjs +1 -1
- package/dist/ui/select.d.cts +11 -11
- package/dist/ui/select.d.mts +11 -11
- package/dist/ui/separator.cjs +1 -1
- package/dist/ui/separator.d.cts +1 -1
- package/dist/ui/separator.d.mts +1 -1
- package/dist/ui/separator.mjs +1 -1
- package/dist/ui/sheet.cjs +1 -1
- package/dist/ui/sheet.d.cts +9 -9
- package/dist/ui/sheet.d.mts +9 -9
- package/dist/ui/sheet.mjs +1 -1
- package/dist/ui/sidebar.cjs +1 -1
- package/dist/ui/sidebar.d.cts +30 -30
- package/dist/ui/sidebar.d.mts +28 -28
- package/dist/ui/sidebar.mjs +1 -1
- package/dist/ui/skeleton.d.cts +2 -2
- package/dist/ui/skeleton.d.mts +2 -2
- package/dist/ui/skeleton.mjs +1 -1
- package/dist/ui/slider.d.cts +2 -2
- package/dist/ui/slider.d.mts +2 -2
- package/dist/ui/sonner.d.cts +2 -2
- package/dist/ui/sonner.d.mts +2 -2
- package/dist/ui/spinner.d.cts +2 -2
- package/dist/ui/spinner.d.mts +2 -2
- package/dist/ui/switch.d.cts +2 -2
- package/dist/ui/switch.d.mts +2 -2
- package/dist/ui/table.d.cts +18 -18
- package/dist/ui/table.d.mts +18 -18
- package/dist/ui/tabs.cjs +1 -1
- package/dist/ui/tabs.d.cts +5 -5
- package/dist/ui/tabs.d.mts +5 -5
- package/dist/ui/tabs.mjs +1 -1
- package/dist/ui/textarea.d.cts +2 -2
- package/dist/ui/textarea.d.mts +2 -2
- package/dist/ui/toggle-group.d.cts +3 -3
- package/dist/ui/toggle-group.d.mts +3 -3
- package/dist/ui/toggle.d.cts +4 -4
- package/dist/ui/toggle.d.mts +2 -2
- package/dist/ui/tooltip.cjs +1 -1
- package/dist/ui/tooltip.d.cts +1 -1
- package/dist/ui/tooltip.d.mts +1 -1
- package/dist/ui/tooltip.mjs +1 -1
- package/package.json +9 -1
- package/dist/date-BUKVV2NZ.cjs.map +0 -1
- package/dist/date-DtgYKyyk.mjs.map +0 -1
- package/dist/layouts/flex.cjs.map +0 -1
- package/dist/layouts/flex.mjs.map +0 -1
- package/dist/typography/paragraph.cjs.map +0 -1
- package/dist/typography/paragraph.mjs.map +0 -1
- package/dist/ui/resizable.cjs.map +0 -1
- package/dist/ui/resizable.mjs.map +0 -1
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{r as e,t}from"../../card-BTswGEgi.mjs";import"../../button.variants-CXNnQvXs.mjs";import{t as n}from"../../button-CqxqZbcQ.mjs";import"../../skeleton-C02GuVQz.mjs";import{t as r}from"../../image-iVki2lFP.mjs";import{i,n as a,r as o,t as s}from"../../tooltip-B7cSlbI6.mjs";import{n as c,r as l,t as u}from"../../avatar-Cp9PebV0.mjs";import"../../close-Bb5Pi28f.mjs";import{a as d,l as f,o as p,r as m,t as h}from"../../dialog-D5bBgIs8.mjs";import{t as g}from"../../input-D-Wrz2-9.mjs";import{t as _}from"../../separator-CZM2q-Ac.mjs";import{c as v,i as y,o as b,r as x,s as S,t as C}from"../../sheet-CUMpo5F_.mjs";import{a as w,l as T,o as E,r as D,t as O}from"../../drawer-D8hxCiVo.mjs";import{a as ee,h as te,o as ne,r as re,t as ie,u as ae}from"../../dropdown-menu-Bt3WFMXE.mjs";import{i as k,n as A,r as oe,t as se}from"../../tabs-BFHJXVlX.mjs";import{t as ce}from"../../scroll-area-CROqtJiH.mjs";import{cn as j}from"@customafk/react-toolkit/utils";import{Fragment as le,jsx as M,jsxs as N}from"react/jsx-runtime";import{LockIcon as ue,LogInIcon as P,LogOutIcon as F,MenuIcon as de,Minus as fe,Plus as pe,ShoppingBag as me,ShoppingBasketIcon as he,ShoppingCartIcon as I,Trash2 as ge,UserIcon as _e}from"lucide-react";import{createContext as L,use as R,useCallback as z,useEffect as B,useMemo as ve,useState as V}from"react";import{cva as ye}from"class-variance-authority";import{Slot as H}from"radix-ui";import{useIsMobile as be}from"@customafk/react-toolkit/hooks/useMobile";import{useMediaQuery as U}from"@customafk/react-toolkit/hooks/useMediaQuery";import{GoogleLogin as W}from"@react-oauth/google";import{useDebounceCallback as xe}from"@customafk/react-toolkit/hooks/useDebounceCallback";const G=L(null),K=()=>{let e=R(G);if(!e)throw Error(`useServiceLayoutContext must be used within a ServiceLayoutProvider`);return e},Se=({id:i,type:a,productName:o,imageUrl:s,quantity:c,price:l,options:u})=>{let{onDeletingCart:d,onUpdatingCart:f}=K(),[p,m]=V(c),[h,_]=V(!1),v=xe(z(()=>{f?.(i,p,a)},[f,i,p,a]),500),y=z(e=>{e<1&&(e=1),e>99&&(e=99),m(e)},[]),b=z(async()=>{_(!0),await d?.(i),_(!1)},[i,d]);return B(()=>(p!==c&&v(),()=>{v.cancel()}),[p,c,v]),N(t,{className:`border-border-weak relative mb-3 overflow-x-auto border p-4 shadow-none`,children:[h&&M(`div`,{className:`bg-muted-muted/80 absolute inset-0 z-10 flex items-center justify-center`,children:M(`div`,{className:`loader-dots`})}),N(e,{className:`p-0`,children:[N(n,{variant:`ghost`,size:`icon`,color:`muted`,disabled:h,className:`text-text-positive-weak absolute top-2 right-2 z-10`,onClick:b,children:[M(ge,{className:`h-4 w-4`}),M(`span`,{className:`sr-only`,children:`Remove item`})]}),N(`div`,{className:`flex gap-3`,children:[M(`div`,{className:`relative size-20 flex-shrink-0`,children:s?M(r,{src:s,alt:o,className:`rounded-md border-none shadow-xs`,width:80,height:80}):M(`div`,{className:`bg-muted text-muted-foreground flex size-full items-center justify-center text-xs`,children:`No image`})}),N(`div`,{className:`flex flex-1 flex-col`,children:[M(`div`,{className:`flex justify-between`,children:M(`h3`,{className:`text-text-positive line-clamp-1 text-sm font-medium`,children:o})}),M(`p`,{className:`text-text-positive-weak mb-1 text-xs`,children:u.map((e,t)=>N(`span`,{children:[e.label,`: `,e.value,t<u.length-1&&`, `]},e.label))}),N(`div`,{className:`mt-auto flex items-center justify-between`,children:[N(`div`,{className:`flex items-center space-x-1`,children:[M(`button`,{disabled:p<=1,className:`border-border active:bg-muted-muted flex size-6 cursor-pointer items-center justify-center rounded-full border transition-all disabled:opacity-60`,onClick:()=>y(p-1),children:M(fe,{size:12})}),M(g,{value:p,onChange:e=>y(parseInt(e.target.value||`1`)),className:`border-border h-6.5 w-14 rounded-md border p-1 text-center text-sm`,min:1}),M(`button`,{disabled:p>=99,className:`border-border active:bg-muted-muted flex size-6 cursor-pointer items-center justify-center rounded-full border transition-all disabled:opacity-60`,onClick:()=>y(p+1),children:M(pe,{size:12})})]}),N(`div`,{className:`text-sm font-semibold`,children:[(l*p).toLocaleString(),` ₫`]})]})]})]})]})]})},q=({items:e=[],cartType:t,className:r})=>{let i=e.reduce((e,t)=>e+t.price*t.quantity,0);return e.length===0?N(`div`,{className:j(`bg-muted-muted flex size-full max-h-80 flex-col items-center justify-center rounded-lg p-8`,r),children:[M(`div`,{className:`text-text-positive-weak bg-card shadow-card mb-4 flex size-20 items-center justify-center rounded-full text-5xl`,children:M(he,{size:42,strokeWidth:1})}),N(`div`,{className:`flex flex-col space-y-1`,children:[M(`p`,{className:`text-text-positive text-center text-base font-semibold`,children:`Giỏ hàng trống`}),M(`p`,{className:`text-text-positive-weak mb-4 text-center text-sm`,children:t===`in_stock`?`Thêm sản phẩm có sẵn vào giỏ hàng của bạn!`:`Thêm sản phẩm đặt trước vào giỏ hàng của bạn!`})]})]}):N(`div`,{className:j(`flex h-full flex-col space-y-4 overflow-y-auto`,r),children:[M(ce,{className:`h-full flex-1`,children:e.map(e=>M(Se,{...e,type:t},e.id))}),N(`div`,{className:`flex-0 space-y-3`,children:[N(`div`,{className:`flex justify-between`,children:[M(`span`,{className:`text-text-positive-weak text-sm`,children:`Tạm tính:`}),N(`span`,{className:`text-sm`,children:[i.toLocaleString(),` ₫`]})]}),M(_,{}),N(`div`,{className:`flex justify-between`,children:[M(`span`,{className:`text-text-positive font-medium`,children:`Tổng cộng:`}),N(`span`,{className:`text-lg font-semibold`,children:[i.toLocaleString(),` ₫`]})]}),M(n,{className:`mt-2 w-full`,children:`Thanh toán ngay`}),t===`pre_order`&&M(`p`,{className:`text-text-positive-weak text-center text-xs italic`,children:`* Sản phẩm đặt trước sẽ được giao sau khi có hàng`})]})]})},J=L(null),Y=()=>{let e=R(J);if(!e)throw Error(`useSidebar must be used within a SidebarProvider.`);return e};function Ce({defaultOpen:e=!0,open:t,onOpenChange:n,className:r,style:i,children:a,...s}){let c=be(),[l,u]=V(!1),[d,f]=V(e),p=t??d,m=z(e=>{let t=typeof e==`function`?e(p):e;n?n(t):f(t),document.cookie=`sidebar_state=${t}; path=/; max-age=604800`},[n,p]),h=z(()=>c?u(e=>!e):m(e=>!e),[c,m]);B(()=>{let e=e=>{e.key===`b`&&(e.metaKey||e.ctrlKey)&&(e.preventDefault(),h())};return window.addEventListener(`keydown`,e),()=>window.removeEventListener(`keydown`,e)},[h]);let g=p?`expanded`:`collapsed`,_=ve(()=>({state:g,isMobile:c,toggleSidebar:h,open:p,setOpen:m,openMobile:l,setOpenMobile:u}),[g,p,m,c,l,h]);return M(J.Provider,{value:_,children:M(o,{delayDuration:0,children:M(`div`,{"data-slot":`sidebar-wrapper`,style:{"--sidebar-width":`16rem`,"--sidebar-width-icon":`3rem`,...i},className:j(`group/sidebar-wrapper`,`has-data-[variant=inset]:bg-sidebar`,`flex h-dvh w-full`,r),...s,children:a})})})}function we({side:e=`left`,variant:t=`sidebar`,collapsible:n=`offcanvas`,className:r,children:i,...a}){let{isMobile:o,state:s,openMobile:c,setOpenMobile:l}=Y();return n===`none`?M(`aside`,{"data-slot":`sidebar`,className:j(`flex h-full w-(--sidebar-width) flex-col bg-sidebar text-sidebar-foreground`,r),...a,children:i}):o?M(C,{open:c,onOpenChange:l,children:N(x,{side:`left`,"data-sidebar":`sidebar`,"data-slot":`sidebar`,"data-mobile":`true`,className:`w-(--sidebar-width) bg-sidebar p-0 text-sidebar-foreground sm:max-w-3xs [&>button]:hidden`,style:{"--sidebar-width":`16rem`},children:[N(b,{className:`sr-only`,children:[M(S,{children:`Sidebar`}),M(y,{children:`Displays the mobile sidebar.`})]}),N(`div`,{className:`flex size-full flex-col`,children:[N(`div`,{className:`flex flex-0 items-center gap-x-2 border-border-weak border-b p-2 pr-4`,children:[M(X,{}),M(`div`,{className:`ml-2 flex aspect-square size-8 items-center justify-center rounded-lg bg-sidebar-primary text-sidebar-primary-foreground`,children:M(I,{size:20})}),N(`div`,{className:`grid flex-1 text-left text-sm leading-tight`,children:[M(`span`,{className:`truncate font-medium`,children:`Lunas Store`}),M(`span`,{className:`truncate text-xs`,children:`Established 2023`})]})]}),M(`div`,{className:`flex flex-1 flex-col p-2`,children:i})]})]})}):N(`aside`,{className:`group peer hidden bg-card text-sidebar-foreground md:block`,"data-state":s,"data-collapsible":s===`collapsed`?n:``,"data-variant":t,"data-side":e,"data-slot":`sidebar`,children:[M(`div`,{"data-slot":`sidebar-gap`,className:j(`relative`,`bg-transparent`,`transition-[width] duration-200 ease-linear`,`h-(--header-height) w-(--sidebar-width)`,`sm:h-[calc(var(--header-height)+0.5rem)]`,`group-data-[collapsible=offcanvas]:w-0`,`group-data-[side=right]:rotate-180`,t===`floating`||t===`inset`?`group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4)))]`:`group-data-[collapsible=icon]:w-(--sidebar-width-icon)`)}),M(`div`,{"data-slot":`sidebar-container`,className:j(`hidden md:flex`,`fixed inset-y-0 top-14 z-10 shadow-nav`,`h-[calc(100dvh-3.5rem)] w-(--sidebar-width)`,`transition-[left,right,width] duration-200 ease-linear`,e===`left`&&`left-0 group-data-[collapsible=offcanvas]:left-[calc(var(--sidebar-width)*-1)]`,e===`right`&&`right-0 group-data-[collapsible=offcanvas]:right-[calc(var(--sidebar-width)*-1)]`,t===`floating`||t===`inset`?`p-2 group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4)))]`:`group-data-[collapsible=icon]:w-(--sidebar-width-icon) group-data-[side=left]:border-r group-data-[side=right]:border-l`,r),...a,children:M(`div`,{"data-sidebar":`sidebar`,"data-slot":`sidebar-inner`,className:j(`flex size-full flex-col`,`group-data-[variant=floating]:rounded-lg`,`group-data-[variant=floating]:border`,`group-data-[variant=floating]:border-sidebar-border`,`group-data-[variant=floating]:shadow-sm`),children:i})})]})}function X({className:e,onClick:t,...r}){let{toggleSidebar:i}=Y();return N(n,{"data-sidebar":`trigger`,"data-slot":`sidebar-trigger`,variant:`ghost`,color:`muted`,size:`icon`,className:j(`size-10 rounded-full`,e),onClick:e=>{t?.(e),i()},...r,children:[M(de,{className:`size-6!`}),M(`span`,{className:`sr-only`,children:`Toggle Sidebar`})]})}function Te({className:e,children:t,...n}){return N(`main`,{"data-slot":`sidebar-inset`,className:j(`relative flex w-full flex-1 flex-col`,e),...n,children:[M(`div`,{className:`h-(--header-height) w-full sm:h-[calc(var(--header-height)+0.5rem)]`}),M(`div`,{className:j(`inset-shadow-sm flex-1`),children:t})]})}function Ee({className:e,children:t,...n}){let{open:r}=Y(),{onLogout:i}=K();return N(`div`,{"data-slot":`sidebar-footer`,"data-sidebar":`footer`,className:j(`flex flex-col gap-2`,e),...n,children:[t,N(Z,{children:[M(Q,{children:N(Me,{className:`border border-border`,onClick:i,children:[M(F,{className:`text-text-positive-weak`}),`Đăng xuất`]})}),r&&M(Q,{className:`mt-2 border-t border-t-border`,children:M(`p`,{className:`pt-2 text-center text-muted-foreground text-xs`,children:`Copyright © 2025, Lunas.`})})]})]})}function De({className:e,...t}){return M(`div`,{"data-slot":`sidebar-content`,"data-sidebar":`content`,className:j(`flex min-h-0 flex-1 flex-col gap-2 overflow-auto group-data-[collapsible=icon]:overflow-hidden`,e),...t})}function Oe({className:e,...t}){return M(`div`,{"data-slot":`sidebar-group`,"data-sidebar":`group`,className:j(`relative flex w-full min-w-0 flex-col`,e),...t})}function ke({className:e,asChild:t=!1,...n}){return M(t?H.Slot:`div`,{"data-slot":`sidebar-group-label`,"data-sidebar":`group-label`,className:j(`flex h-8 shrink-0 items-center rounded-md px-2 font-medium text-xs outline-hidden transition-[margin,opacity] duration-200 ease-linear focus-visible:ring-2`,`text-sidebar-foreground/70`,`ring-sidebar-ring`,`[&>svg]:size-4`,`[&>svg]:shrink-0`,`group-data-[collapsible=icon]:-mt-8`,`group-data-[collapsible=icon]:opacity-0`,e),...n})}function Ae({className:e,...t}){return M(`div`,{"data-slot":`sidebar-group-content`,"data-sidebar":`group-content`,className:j(`w-full text-sm`,e),...t})}function Z({className:e,...t}){return M(`ul`,{"data-slot":`sidebar-menu`,"data-sidebar":`menu`,className:j(`flex w-full min-w-0 flex-col gap-1`,e),...t})}function Q({className:e,...t}){return M(`li`,{"data-slot":`sidebar-menu-item`,"data-sidebar":`menu-item`,className:j(`group/menu-item relative`,e),...t})}const je=ye(`peer/menu-button.cursor-pointer.flex w-full items-center gap-2.overflow-hidden rounded-md p-2 outline-hidden.truncate text-left.transition-[color,width,height,padding].hover:bg-sidebar-accent.hover:text-sidebar-accent-foreground.active:bg-sidebar-accent.active:text-sidebar-accent-foreground.disabled:pointer-events-none.disabled:opacity-50.group-has-data-[sidebar=menu-action]/menu-item:pr-8.aria-disabled:pointer-events-none.aria-disabled:opacity-50.data-[active=true]:bg-sidebar-primary-muted.data-[active=true]:font-medium.data-[active=true]:text-sidebar-primary.data-[state=open]:hover:bg-sidebar-accent.data-[state=open]:hover:text-sidebar-accent-foreground.group-data-[collapsible=icon]:size-12!.group-data-[collapsible=icon]:p-3!.group-data-[collapsible=icon]:gap-3!.[&>svg]:size-6.[&>svg]:shrink-0.[&>span:last-child]:truncate`.split(`.`),{variants:{variant:{default:`hover:bg-sidebar-accent/60 hover:text-sidebar-accent-foreground/80`,outline:`bg-background shadow-[0_0_0_1px_hsl(var(--sidebar-border))] hover:bg-sidebar-accent hover:text-sidebar-accent-foreground hover:shadow-[0_0_0_1px_hsl(var(--sidebar-accent))]`},size:{default:`h-10 text-sm`,sm:`h-7 text-xs`,lg:`h-12 text-sm group-data-[collapsible=icon]:p-0!`}},defaultVariants:{variant:`default`,size:`default`}});function Me({asChild:e=!1,isActive:t=!1,variant:n=`default`,size:r=`default`,tooltip:o,className:c,...l}){let u=e?H.Slot:`button`,{isMobile:d,state:f}=Y(),p=M(u,{"data-slot":`sidebar-menu-button`,"data-sidebar":`menu-button`,"data-size":r,"data-active":t,className:j(je({variant:n,size:r}),c),...l});return o?(typeof o==`string`&&(o={children:o}),N(s,{children:[M(i,{asChild:!0,children:p}),M(a,{side:`right`,align:`center`,hidden:f!==`collapsed`||d,...o})]})):p}const Ne=({isLoggedIn:e=!1,username:t,email:n,inStockCarts:r,orderedCarts:i,onGoogleLoginSuccess:a,onUpdatingCart:o,onDeletingCart:s,onLogout:c,children:l})=>M(G.Provider,{value:{isLoggedIn:e,username:t,email:n,inStockCarts:r,orderedCarts:i,onGoogleLoginSuccess:a,onUpdatingCart:o,onDeletingCart:s,onLogout:c},children:l}),Pe=({children:e})=>M(Ce,{children:e}),Fe=({userName:e=`Keith Kennedy`,userEmail:t=`k.kennedy@originui.com`,onLogout:r})=>N(ie,{children:[M(te,{asChild:!0,children:M(n,{size:`icon`,variant:`ghost`,color:`secondary`,className:`size-10 rounded-full`,children:N(u,{className:`size-10`,children:[M(l,{}),M(c,{className:`size-full bg-muted-muted`,children:M(_e,{})})]})})}),N(re,{align:`end`,className:`max-w-64`,children:[N(ne,{className:`flex min-w-0 flex-col`,children:[M(`span`,{className:`truncate font-medium text-sm text-text-positive`,children:e}),M(`span`,{className:`truncate font-normal text-text-positive-weak text-xs`,children:t})]}),M(ae,{}),N(ee,{onClick:r,children:[M(F,{size:16,"aria-hidden":`true`}),M(`span`,{className:`text-text-positive`,children:`Logout`})]})]})]}),$=()=>{let{inStockCarts:e=[],orderedCarts:t=[]}=K(),r=e.length,i=t.length,a=r+i;return N(C,{children:[M(v,{asChild:!0,children:M(n,{size:`icon`,variant:`ghost`,color:`secondary`,className:`relative size-10 rounded-full`,children:M(I,{})})}),N(x,{className:`w-[95vw] sm:max-w-md`,children:[M(b,{className:`flex-0 border-border-weak border-b pb-3`,children:N(S,{className:`flex items-center gap-2`,children:[M(me,{size:20}),M(`span`,{children:`Giỏ hàng của bạn`}),a>0&&N(`span`,{className:`font-normal text-sm text-text-positive-weak`,children:[`(`,a,` sản phẩm)`]})]})}),N(se,{defaultValue:i>0?`pre_order`:`in_stock`,className:`h-full flex-1 overflow-y-auto p-4 pt-0`,children:[N(oe,{className:`w-full flex-0`,children:[N(k,{value:`in_stock`,className:`relative`,children:[`Có sẵn`,r>0&&N(`span`,{children:[`(`,r,`)`]})]}),N(k,{value:`pre_order`,className:`relative`,children:[`Đặt trước`,i>0&&N(`span`,{children:[`(`,i,`)`]})]})]}),M(A,{value:`pre_order`,className:`flex-1 overflow-y-auto`,children:M(q,{items:t,cartType:`pre_order`})}),M(A,{value:`in_stock`,className:`flex-1 overflow-y-auto`,children:M(q,{items:e,cartType:`in_stock`})})]})]})]})},Ie=()=>{let{isLoggedIn:e,username:t,email:r,onGoogleLoginSuccess:i,onLogout:a}=K(),o=U(`(min-width: 640px)`),[s,c]=V(!1);return N(`header`,{className:j(`bg-card`,`h-(--header-height)`,`sm:h-[calc(var(--header-height)+0.5rem)] sm:px-4 sm:pr-6`,`absolute inset-x-0 top-0 z-20 gap-2 px-2 pr-4.5`,`flex items-center shadow-nav`,`transition-[width,height] ease-linear`),children:[e&&M(X,{}),N(`div`,{className:`flex gap-x-2 sm:ml-2.5`,children:[M(`div`,{className:`flex aspect-square size-8 items-center justify-center rounded-lg bg-sidebar-primary text-sidebar-primary-foreground`,children:M(I,{size:20})}),N(`div`,{className:`grid flex-1 text-left text-sm leading-tight`,children:[M(`span`,{className:`truncate font-medium`,children:`Lunas Store`}),M(`span`,{className:`truncate text-xs`,children:`Established 2023`})]})]}),N(`div`,{className:`flex flex-1 items-center justify-end gap-x-2`,children:[!e&&N(n,{className:`w-8 sm:w-fit`,onClick:()=>c(!0),children:[M(P,{}),M(`span`,{className:`sr-only sm:not-sr-only`,children:`Đăng nhập`})]}),e&&N(le,{children:[M(Fe,{userName:t,userEmail:r,onLogout:a}),M(_,{orientation:`vertical`,className:`min-h-6 w-px`}),M($,{})]})]}),o&&M(h,{open:s,onOpenChange:c,children:N(m,{showCloseButton:!1,className:`flex flex-col gap-0 border-none p-0 sm:max-w-sm`,children:[M(p,{className:`flex-0 gap-2 p-6`,children:M(f,{className:`text-center`,children:`Chào mừng bạn đến với Lunas Store!`})}),M(`div`,{className:`flex flex-1 flex-col`,children:M(`main`,{className:`size-full flex-1 bg-card p-4 pt-0`,children:N(`div`,{className:`flex flex-col items-center gap-y-1`,children:[M(`p`,{className:`text-sm text-text-positive-weak`,children:`Đăng nhập với Google`}),M(W,{size:`large`,theme:`outline`,width:240,onSuccess:async e=>{!e.clientId||!e.credential||!e.select_by||await i?.(e)}})]})})}),M(d,{className:`p-2`})]})}),!o&&M(O,{open:s,onOpenChange:c,children:N(D,{children:[M(E,{className:`text-left`,children:M(T,{children:`Chào mừng bạn đến với Lunas Store!`})}),M(`div`,{className:`flex flex-1 flex-col`,children:M(`main`,{className:`flex size-full flex-1 flex-col p-4 pt-0`,children:N(`div`,{className:`flex flex-col items-center gap-y-1`,children:[M(`p`,{className:`text-sm text-text-positive-weak`,children:`Đăng nhập với Google`}),M(W,{size:`large`,theme:`outline`,onSuccess:async e=>{!e.clientId||!e.credential||!e.select_by||await i?.(e)}})]})})}),M(w,{})]})})]})},Le=({children:e,...t})=>{let{isLoggedIn:n}=K();return n?M(we,{variant:`inset`,collapsible:`icon`,...t,children:e}):null},Re=({children:e})=>{let{isLoggedIn:t,onGoogleLoginSuccess:r}=K(),i=U(`(min-width: 640px)`),[a,o]=V(!1);return t?M(Te,{children:M(`section`,{className:`relative size-full`,children:M(`div`,{className:`absolute inset-0 flex flex-col`,children:e})})}):M(`div`,{className:`size-full p-4 pt-[calc(var(--header-height)+1.5rem)]`,children:N(`div`,{className:`flex size-full flex-col items-center justify-center gap-6 rounded-lg bg-card p-8 text-center shadow-card`,children:[M(`div`,{className:`flex size-20 items-center justify-center rounded-full bg-muted-foreground/10`,children:M(ue,{className:`size-10 text-primary`})}),N(`div`,{className:`flex max-w-md flex-col gap-2`,children:[M(`h2`,{className:`font-semibold text-2xl`,children:`Bạn chưa đăng nhập`}),M(`p`,{className:`text-text-positive-weak`,children:`Đăng nhập để khám phá đầy đủ các tính năng của Lunas Store và truy cập vào tài khoản của bạn.`})]}),N(n,{size:`lg`,className:`gap-2`,onClick:()=>o(!0),children:[M(P,{size:18}),M(`span`,{children:`Đăng nhập ngay`})]}),i&&M(h,{open:a,onOpenChange:o,children:N(m,{showCloseButton:!1,className:`flex flex-col gap-0 border-none p-0 sm:max-w-sm`,children:[M(p,{className:`flex-0 gap-2 p-6`,children:M(f,{className:`text-center`,children:`Chào mừng bạn đến với Lunas Store!`})}),M(`div`,{className:`flex flex-1 flex-col`,children:M(`main`,{className:`size-full flex-1 bg-card p-4 pt-0`,children:N(`div`,{className:`flex flex-col items-center gap-y-1`,children:[M(`p`,{className:`text-sm text-text-positive-weak`,children:`Đăng nhập với Google`}),M(W,{size:`large`,theme:`outline`,width:240,onSuccess:async e=>{!e.clientId||!e.credential||!e.select_by||(await r?.(e),o(!1))}})]})})}),M(d,{className:`p-2`})]})}),!i&&M(O,{open:a,onOpenChange:o,children:N(D,{children:[M(E,{className:`text-left`,children:M(T,{children:`Chào mừng bạn đến với Lunas Store!`})}),M(`div`,{className:`flex flex-1 flex-col`,children:M(`main`,{className:`flex size-full flex-1 flex-col p-4 pt-0`,children:N(`div`,{className:`flex flex-col items-center gap-y-1`,children:[M(`p`,{className:`text-sm text-text-positive-weak`,children:`Đăng nhập với Google`}),M(W,{size:`large`,theme:`outline`,onSuccess:async e=>{!e.clientId||!e.credential||!e.select_by||(await r?.(e),o(!1))}})]})})}),M(w,{})]})})]})})},ze=({className:e,children:t})=>M(`div`,{"data-slot":`main-header`,className:j(`flex-0 snap-start`,e),children:t}),Be=({className:e,children:t})=>M(`div`,{"data-slot":`main-content`,className:j(`flex w-full flex-1 flex-col gap-4 overflow-y-auto px-2 sm:px-4`,e),children:t}),Ve=({className:e,children:t})=>M(`div`,{"data-slot":`main-footer`,className:j(`hidden w-full flex-0 border-border-weak border-t pt-2 sm:flex`,e),children:t}),He=({children:e,className:t})=>M(`div`,{"data-slot":`main-group`,className:j(`flex size-full flex-col gap-4`,t),children:e}),Ue=({className:e,children:t})=>M(`div`,{"data-slot":`main-group-content`,className:j(`size-full max-w-8xl flex-1 rounded-md bg-card p-4 shadow-card`,e),children:t});export{$ as ServiceLayoutCartInfo,Ie as ServiceLayoutHeader,Re as ServiceLayoutMain,Be as ServiceLayoutMainContent,Ve as ServiceLayoutMainFooter,He as ServiceLayoutMainGroup,Ue as ServiceLayoutMainGroupContent,ze as ServiceLayoutMainHeader,Ne as ServiceLayoutProvider,Le as ServiceLayoutSidebar,De as ServiceLayoutSidebarContent,Ee as ServiceLayoutSidebarFooter,Oe as ServiceLayoutSidebarGroup,Ae as ServiceLayoutSidebarGroupContent,ke as ServiceLayoutSidebarGroupLabel,Z as ServiceLayoutSidebarMenu,Me as ServiceLayoutSidebarMenuButton,Q as ServiceLayoutSidebarMenuItem,Fe as ServiceLayoutUserInfo,Pe as ServiceLayoutWrapper};
|
|
1
|
+
import{r as e,t}from"../../card-jwcpmq7e.mjs";import"../../button.variants-DmUzMbB8.mjs";import{t as n}from"../../button-DIEXNBpi.mjs";import"../../skeleton-CvZncisJ.mjs";import{t as r}from"../../image-BwbAQRxx.mjs";import{i,n as a,r as o,t as s}from"../../tooltip-D0XDvxqi.mjs";import{n as c,r as l,t as u}from"../../avatar-BaxX-xI9.mjs";import"../../close-CGei_VQ6.mjs";import{a as d,l as f,o as p,r as m,t as h}from"../../dialog-DeZVOOmh.mjs";import{t as g}from"../../input-CC10BdKh.mjs";import{t as _}from"../../separator-C2yOD6r_.mjs";import{c as v,i as y,o as b,r as x,s as S,t as C}from"../../sheet-DXLayspn.mjs";import{a as ee,h as te,o as ne,r as re,t as ie,u as ae}from"../../dropdown-menu-CtU6oBI4.mjs";import{a as w,l as T,o as oe,r as E,t as D}from"../../drawer-272Fecsq.mjs";import{i as O,n as k,r as se,t as ce}from"../../tabs-Cjh2jtUZ.mjs";import{t as le}from"../../scroll-area-CiMS2UY7.mjs";import{cn as A}from"@customafk/react-toolkit/utils";import{Fragment as ue,jsx as j,jsxs as M}from"react/jsx-runtime";import{LockIcon as de,LogInIcon as N,LogOutIcon as P,MenuIcon as fe,Minus as pe,Plus as me,ShoppingBag as he,ShoppingBasketIcon as ge,ShoppingCartIcon as F,Trash2 as _e,UserIcon as ve}from"lucide-react";import{createContext as I,use as L,useCallback as R,useEffect as z,useMemo as ye,useState as B}from"react";import{cva as be}from"class-variance-authority";import{Slot as V}from"radix-ui";import{useIsMobile as xe}from"@customafk/react-toolkit/hooks/useMobile";import{useMediaQuery as H}from"@customafk/react-toolkit/hooks/useMediaQuery";import{GoogleLogin as U}from"@react-oauth/google";import{useDebounceCallback as Se}from"@customafk/react-toolkit/hooks/useDebounceCallback";const W=I(null),G=()=>{let e=L(W);if(!e)throw Error(`useServiceLayoutContext must be used within a ServiceLayoutProvider`);return e},Ce=({id:i,type:a,productName:o,imageUrl:s,quantity:c,price:l,options:u})=>{let{onDeletingCart:d,onUpdatingCart:f}=G(),[p,m]=B(c),[h,_]=B(!1),v=Se(R(()=>{f?.(i,p,a)},[f,i,p,a]),500),y=R(e=>{e<1&&(e=1),e>99&&(e=99),m(e)},[]),b=R(async()=>{_(!0),await d?.(i),_(!1)},[i,d]);return z(()=>(p!==c&&v(),()=>{v.cancel()}),[p,c,v]),M(t,{className:`border-border-weak relative mb-3 overflow-x-auto border p-4 shadow-none`,children:[h&&j(`div`,{className:`bg-muted-muted/80 absolute inset-0 z-10 flex items-center justify-center`,children:j(`div`,{className:`loader-dots`})}),M(e,{className:`p-0`,children:[M(n,{variant:`ghost`,size:`icon`,color:`muted`,disabled:h,className:`text-text-positive-weak absolute top-2 right-2 z-10`,onClick:b,children:[j(_e,{className:`h-4 w-4`}),j(`span`,{className:`sr-only`,children:`Remove item`})]}),M(`div`,{className:`flex gap-3`,children:[j(`div`,{className:`relative size-20 flex-shrink-0`,children:s?j(r,{src:s,alt:o,className:`rounded-md border-none shadow-xs`,width:80,height:80}):j(`div`,{className:`bg-muted text-muted-foreground flex size-full items-center justify-center text-xs`,children:`No image`})}),M(`div`,{className:`flex flex-1 flex-col`,children:[j(`div`,{className:`flex justify-between`,children:j(`h3`,{className:`text-text-positive line-clamp-1 text-sm font-medium`,children:o})}),j(`p`,{className:`text-text-positive-weak mb-1 text-xs`,children:u.map((e,t)=>M(`span`,{children:[e.label,`: `,e.value,t<u.length-1&&`, `]},e.label))}),M(`div`,{className:`mt-auto flex items-center justify-between`,children:[M(`div`,{className:`flex items-center space-x-1`,children:[j(`button`,{disabled:p<=1,className:`border-border active:bg-muted-muted flex size-6 cursor-pointer items-center justify-center rounded-full border transition-all disabled:opacity-60`,onClick:()=>y(p-1),children:j(pe,{size:12})}),j(g,{value:p,onChange:e=>y(parseInt(e.target.value||`1`)),className:`border-border h-6.5 w-14 rounded-md border p-1 text-center text-sm`,min:1}),j(`button`,{disabled:p>=99,className:`border-border active:bg-muted-muted flex size-6 cursor-pointer items-center justify-center rounded-full border transition-all disabled:opacity-60`,onClick:()=>y(p+1),children:j(me,{size:12})})]}),M(`div`,{className:`text-sm font-semibold`,children:[(l*p).toLocaleString(),` ₫`]})]})]})]})]})]})},K=({items:e=[],cartType:t,className:r})=>{let i=e.reduce((e,t)=>e+t.price*t.quantity,0);return e.length===0?M(`div`,{className:A(`bg-muted-muted flex size-full max-h-80 flex-col items-center justify-center rounded-lg p-8`,r),children:[j(`div`,{className:`text-text-positive-weak bg-card shadow-card mb-4 flex size-20 items-center justify-center rounded-full text-5xl`,children:j(ge,{size:42,strokeWidth:1})}),M(`div`,{className:`flex flex-col space-y-1`,children:[j(`p`,{className:`text-text-positive text-center text-base font-semibold`,children:`Giỏ hàng trống`}),j(`p`,{className:`text-text-positive-weak mb-4 text-center text-sm`,children:t===`in_stock`?`Thêm sản phẩm có sẵn vào giỏ hàng của bạn!`:`Thêm sản phẩm đặt trước vào giỏ hàng của bạn!`})]})]}):M(`div`,{className:A(`flex h-full flex-col space-y-4 overflow-y-auto`,r),children:[j(le,{className:`h-full flex-1`,children:e.map(e=>j(Ce,{...e,type:t},e.id))}),M(`div`,{className:`flex-0 space-y-3`,children:[M(`div`,{className:`flex justify-between`,children:[j(`span`,{className:`text-text-positive-weak text-sm`,children:`Tạm tính:`}),M(`span`,{className:`text-sm`,children:[i.toLocaleString(),` ₫`]})]}),j(_,{}),M(`div`,{className:`flex justify-between`,children:[j(`span`,{className:`text-text-positive font-medium`,children:`Tổng cộng:`}),M(`span`,{className:`text-lg font-semibold`,children:[i.toLocaleString(),` ₫`]})]}),j(n,{className:`mt-2 w-full`,children:`Thanh toán ngay`}),t===`pre_order`&&j(`p`,{className:`text-text-positive-weak text-center text-xs italic`,children:`* Sản phẩm đặt trước sẽ được giao sau khi có hàng`})]})]})},q=I(null),J=()=>{let e=L(q);if(!e)throw Error(`useSidebar must be used within a SidebarProvider.`);return e};function we({defaultOpen:e=!0,open:t,onOpenChange:n,className:r,style:i,children:a,...s}){let c=xe(),[l,u]=B(!1),[d,f]=B(e),p=t??d,m=R(e=>{let t=typeof e==`function`?e(p):e;n?n(t):f(t),document.cookie=`sidebar_state=${t}; path=/; max-age=604800`},[n,p]),h=R(()=>c?u(e=>!e):m(e=>!e),[c,m]);z(()=>{let e=e=>{e.key===`b`&&(e.metaKey||e.ctrlKey)&&(e.preventDefault(),h())};return window.addEventListener(`keydown`,e),()=>window.removeEventListener(`keydown`,e)},[h]);let g=p?`expanded`:`collapsed`,_=ye(()=>({state:g,isMobile:c,toggleSidebar:h,open:p,setOpen:m,openMobile:l,setOpenMobile:u}),[g,p,m,c,l,h]);return j(q.Provider,{value:_,children:j(o,{delayDuration:0,children:j(`div`,{"data-slot":`sidebar-wrapper`,style:{"--sidebar-width":`16rem`,"--sidebar-width-icon":`3rem`,...i},className:A(`group/sidebar-wrapper`,`has-data-[variant=inset]:bg-sidebar`,`flex h-dvh w-full`,r),...s,children:a})})})}function Te({side:e=`left`,variant:t=`sidebar`,collapsible:n=`offcanvas`,className:r,children:i,...a}){let{isMobile:o,state:s,openMobile:c,setOpenMobile:l}=J();return n===`none`?j(`aside`,{"data-slot":`sidebar`,className:A(`flex h-full w-(--sidebar-width) flex-col bg-sidebar text-sidebar-foreground`,r),...a,children:i}):o?j(C,{open:c,onOpenChange:l,children:M(x,{side:`left`,"data-sidebar":`sidebar`,"data-slot":`sidebar`,"data-mobile":`true`,className:`w-(--sidebar-width) bg-sidebar p-0 text-sidebar-foreground sm:max-w-3xs [&>button]:hidden`,style:{"--sidebar-width":`16rem`},children:[M(b,{className:`sr-only`,children:[j(S,{children:`Sidebar`}),j(y,{children:`Displays the mobile sidebar.`})]}),M(`div`,{className:`flex size-full flex-col`,children:[M(`div`,{className:`flex flex-0 items-center gap-x-2 border-border-weak border-b p-2 pr-4`,children:[j(Y,{}),j(`div`,{className:`ml-2 flex aspect-square size-8 items-center justify-center rounded-lg bg-sidebar-primary text-sidebar-primary-foreground`,children:j(F,{size:20})}),M(`div`,{className:`grid flex-1 text-left text-sm leading-tight`,children:[j(`span`,{className:`truncate font-medium`,children:`Lunas Store`}),j(`span`,{className:`truncate text-xs`,children:`Established 2023`})]})]}),j(`div`,{className:`flex flex-1 flex-col p-2`,children:i})]})]})}):M(`aside`,{className:`group peer hidden bg-card text-sidebar-foreground md:block`,"data-state":s,"data-collapsible":s===`collapsed`?n:``,"data-variant":t,"data-side":e,"data-slot":`sidebar`,children:[j(`div`,{"data-slot":`sidebar-gap`,className:A(`relative`,`bg-transparent`,`transition-[width] duration-200 ease-linear`,`h-(--header-height) w-(--sidebar-width)`,`sm:h-[calc(var(--header-height)+0.5rem)]`,`group-data-[collapsible=offcanvas]:w-0`,`group-data-[side=right]:rotate-180`,t===`floating`||t===`inset`?`group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4)))]`:`group-data-[collapsible=icon]:w-(--sidebar-width-icon)`)}),j(`div`,{"data-slot":`sidebar-container`,className:A(`hidden md:flex`,`fixed inset-y-0 top-14 z-10 shadow-nav`,`h-[calc(100dvh-3.5rem)] w-(--sidebar-width)`,`transition-[left,right,width] duration-200 ease-linear`,e===`left`&&`left-0 group-data-[collapsible=offcanvas]:left-[calc(var(--sidebar-width)*-1)]`,e===`right`&&`right-0 group-data-[collapsible=offcanvas]:right-[calc(var(--sidebar-width)*-1)]`,t===`floating`||t===`inset`?`p-2 group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4)))]`:`group-data-[collapsible=icon]:w-(--sidebar-width-icon) group-data-[side=left]:border-r group-data-[side=right]:border-l`,r),...a,children:j(`div`,{"data-sidebar":`sidebar`,"data-slot":`sidebar-inner`,className:A(`flex size-full flex-col`,`group-data-[variant=floating]:rounded-lg`,`group-data-[variant=floating]:border`,`group-data-[variant=floating]:border-sidebar-border`,`group-data-[variant=floating]:shadow-sm`),children:i})})]})}function Y({className:e,onClick:t,...r}){let{toggleSidebar:i}=J();return M(n,{"data-sidebar":`trigger`,"data-slot":`sidebar-trigger`,variant:`ghost`,color:`muted`,size:`icon`,className:A(`size-10 rounded-full`,e),onClick:e=>{t?.(e),i()},...r,children:[j(fe,{className:`size-6!`}),j(`span`,{className:`sr-only`,children:`Toggle Sidebar`})]})}function Ee({className:e,children:t,...n}){return M(`main`,{"data-slot":`sidebar-inset`,className:A(`relative flex w-full flex-1 flex-col`,e),...n,children:[j(`div`,{className:`h-(--header-height) w-full sm:h-[calc(var(--header-height)+0.5rem)]`}),j(`div`,{className:A(`inset-shadow-sm flex-1`),children:t})]})}function De({className:e,children:t,...n}){let{open:r}=J(),{onLogout:i}=G();return M(`div`,{"data-slot":`sidebar-footer`,"data-sidebar":`footer`,className:A(`flex flex-col gap-2`,e),...n,children:[t,M(X,{children:[j(Z,{children:M(Q,{className:`border border-border`,onClick:i,children:[j(P,{className:`text-text-positive-weak`}),`Đăng xuất`]})}),r&&j(Z,{className:`mt-2 border-t border-t-border`,children:j(`p`,{className:`pt-2 text-center text-muted-foreground text-xs`,children:`Copyright © 2025, Lunas.`})})]})]})}function Oe({className:e,...t}){return j(`div`,{"data-slot":`sidebar-content`,"data-sidebar":`content`,className:A(`flex min-h-0 flex-1 flex-col gap-2 overflow-auto group-data-[collapsible=icon]:overflow-hidden`,e),...t})}function ke({className:e,...t}){return j(`div`,{"data-slot":`sidebar-group`,"data-sidebar":`group`,className:A(`relative flex w-full min-w-0 flex-col`,e),...t})}function Ae({className:e,asChild:t=!1,...n}){return j(t?V.Slot:`div`,{"data-slot":`sidebar-group-label`,"data-sidebar":`group-label`,className:A(`flex h-8 shrink-0 items-center rounded-md px-2 font-medium text-xs outline-hidden transition-[margin,opacity] duration-200 ease-linear focus-visible:ring-2`,`text-sidebar-foreground/70`,`ring-sidebar-ring`,`[&>svg]:size-4`,`[&>svg]:shrink-0`,`group-data-[collapsible=icon]:-mt-8`,`group-data-[collapsible=icon]:opacity-0`,e),...n})}function je({className:e,...t}){return j(`div`,{"data-slot":`sidebar-group-content`,"data-sidebar":`group-content`,className:A(`w-full text-sm`,e),...t})}function X({className:e,...t}){return j(`ul`,{"data-slot":`sidebar-menu`,"data-sidebar":`menu`,className:A(`flex w-full min-w-0 flex-col gap-1`,e),...t})}function Z({className:e,...t}){return j(`li`,{"data-slot":`sidebar-menu-item`,"data-sidebar":`menu-item`,className:A(`group/menu-item relative`,e),...t})}const Me=be(`peer/menu-button.cursor-pointer.flex w-full items-center gap-2.overflow-hidden rounded-md p-2 outline-hidden.truncate text-left.transition-[color,width,height,padding].hover:bg-sidebar-accent.hover:text-sidebar-accent-foreground.active:bg-sidebar-accent.active:text-sidebar-accent-foreground.disabled:pointer-events-none.disabled:opacity-50.group-has-data-[sidebar=menu-action]/menu-item:pr-8.aria-disabled:pointer-events-none.aria-disabled:opacity-50.data-[active=true]:bg-sidebar-primary-muted.data-[active=true]:font-medium.data-[active=true]:text-sidebar-primary.data-[state=open]:hover:bg-sidebar-accent.data-[state=open]:hover:text-sidebar-accent-foreground.group-data-[collapsible=icon]:size-12!.group-data-[collapsible=icon]:p-3!.group-data-[collapsible=icon]:gap-3!.[&>svg]:size-6.[&>svg]:shrink-0.[&>span:last-child]:truncate`.split(`.`),{variants:{variant:{default:`hover:bg-sidebar-accent/60 hover:text-sidebar-accent-foreground/80`,outline:`bg-background shadow-[0_0_0_1px_hsl(var(--sidebar-border))] hover:bg-sidebar-accent hover:text-sidebar-accent-foreground hover:shadow-[0_0_0_1px_hsl(var(--sidebar-accent))]`},size:{default:`h-10 text-sm`,sm:`h-7 text-xs`,lg:`h-12 text-sm group-data-[collapsible=icon]:p-0!`}},defaultVariants:{variant:`default`,size:`default`}});function Q({asChild:e=!1,isActive:t=!1,variant:n=`default`,size:r=`default`,tooltip:o,className:c,...l}){let u=e?V.Slot:`button`,{isMobile:d,state:f}=J(),p=j(u,{"data-slot":`sidebar-menu-button`,"data-sidebar":`menu-button`,"data-size":r,"data-active":t,className:A(Me({variant:n,size:r}),c),...l});return o?(typeof o==`string`&&(o={children:o}),M(s,{children:[j(i,{asChild:!0,children:p}),j(a,{side:`right`,align:`center`,hidden:f!==`collapsed`||d,...o})]})):p}const Ne=({isLoggedIn:e=!1,username:t,email:n,inStockCarts:r,orderedCarts:i,onGoogleLoginSuccess:a,onUpdatingCart:o,onDeletingCart:s,onLogout:c,children:l})=>j(W.Provider,{value:{isLoggedIn:e,username:t,email:n,inStockCarts:r,orderedCarts:i,onGoogleLoginSuccess:a,onUpdatingCart:o,onDeletingCart:s,onLogout:c},children:l}),Pe=({children:e})=>j(we,{children:e}),$=({userName:e=`Keith Kennedy`,userEmail:t=`k.kennedy@originui.com`,onLogout:r})=>M(ie,{children:[j(te,{asChild:!0,children:j(n,{size:`icon`,variant:`ghost`,color:`secondary`,className:`size-10 rounded-full`,children:M(u,{className:`size-10`,children:[j(l,{}),j(c,{className:`size-full bg-muted-muted`,children:j(ve,{})})]})})}),M(re,{align:`end`,className:`max-w-64`,children:[M(ne,{className:`flex min-w-0 flex-col`,children:[j(`span`,{className:`truncate font-medium text-sm text-text-positive`,children:e}),j(`span`,{className:`truncate font-normal text-text-positive-weak text-xs`,children:t})]}),j(ae,{}),M(ee,{onClick:r,children:[j(P,{size:16,"aria-hidden":`true`}),j(`span`,{className:`text-text-positive`,children:`Logout`})]})]})]}),Fe=()=>{let{inStockCarts:e=[],orderedCarts:t=[]}=G(),r=e.length,i=t.length,a=r+i;return M(C,{children:[j(v,{asChild:!0,children:j(n,{size:`icon`,variant:`ghost`,color:`secondary`,className:`relative size-10 rounded-full`,children:j(F,{})})}),M(x,{className:`w-[95vw] sm:max-w-md`,children:[j(b,{className:`flex-0 border-border-weak border-b pb-3`,children:M(S,{className:`flex items-center gap-2`,children:[j(he,{size:20}),j(`span`,{children:`Giỏ hàng của bạn`}),a>0&&M(`span`,{className:`font-normal text-sm text-text-positive-weak`,children:[`(`,a,` sản phẩm)`]})]})}),M(ce,{defaultValue:i>0?`pre_order`:`in_stock`,className:`h-full flex-1 overflow-y-auto p-4 pt-0`,children:[M(se,{className:`w-full flex-0`,children:[M(O,{value:`in_stock`,className:`relative`,children:[`Có sẵn`,r>0&&M(`span`,{children:[`(`,r,`)`]})]}),M(O,{value:`pre_order`,className:`relative`,children:[`Đặt trước`,i>0&&M(`span`,{children:[`(`,i,`)`]})]})]}),j(k,{value:`pre_order`,className:`flex-1 overflow-y-auto`,children:j(K,{items:t,cartType:`pre_order`})}),j(k,{value:`in_stock`,className:`flex-1 overflow-y-auto`,children:j(K,{items:e,cartType:`in_stock`})})]})]})]})},Ie=()=>{let{isLoggedIn:e,username:t,email:r,onGoogleLoginSuccess:i,onLogout:a}=G(),o=H(`(min-width: 640px)`),[s,c]=B(!1);return M(`header`,{className:A(`bg-card`,`h-(--header-height)`,`sm:h-[calc(var(--header-height)+0.5rem)] sm:px-4 sm:pr-6`,`absolute inset-x-0 top-0 z-20 gap-2 px-2 pr-4.5`,`flex items-center shadow-nav`,`transition-[width,height] ease-linear`),children:[e&&j(Y,{}),M(`div`,{className:`flex gap-x-2 sm:ml-2.5`,children:[j(`div`,{className:`flex aspect-square size-8 items-center justify-center rounded-lg bg-sidebar-primary text-sidebar-primary-foreground`,children:j(F,{size:20})}),M(`div`,{className:`grid flex-1 text-left text-sm leading-tight`,children:[j(`span`,{className:`truncate font-medium`,children:`Lunas Store`}),j(`span`,{className:`truncate text-xs`,children:`Established 2023`})]})]}),M(`div`,{className:`flex flex-1 items-center justify-end gap-x-2`,children:[!e&&M(n,{className:`w-8 sm:w-fit`,onClick:()=>c(!0),children:[j(N,{}),j(`span`,{className:`sr-only sm:not-sr-only`,children:`Đăng nhập`})]}),e&&M(ue,{children:[j($,{userName:t,userEmail:r,onLogout:a}),j(_,{orientation:`vertical`,className:`min-h-6 w-px`}),j(Fe,{})]})]}),o&&j(h,{open:s,onOpenChange:c,children:M(m,{showCloseButton:!1,className:`flex flex-col gap-0 border-none p-0 sm:max-w-sm`,children:[j(p,{className:`flex-0 gap-2 p-6`,children:j(f,{className:`text-center`,children:`Chào mừng bạn đến với Lunas Store!`})}),j(`div`,{className:`flex flex-1 flex-col`,children:j(`main`,{className:`size-full flex-1 bg-card p-4 pt-0`,children:M(`div`,{className:`flex flex-col items-center gap-y-1`,children:[j(`p`,{className:`text-sm text-text-positive-weak`,children:`Đăng nhập với Google`}),j(U,{size:`large`,theme:`outline`,width:240,onSuccess:async e=>{!e.clientId||!e.credential||!e.select_by||await i?.(e)}})]})})}),j(d,{className:`p-2`})]})}),!o&&j(D,{open:s,onOpenChange:c,children:M(E,{children:[j(oe,{className:`text-left`,children:j(T,{children:`Chào mừng bạn đến với Lunas Store!`})}),j(`div`,{className:`flex flex-1 flex-col`,children:j(`main`,{className:`flex size-full flex-1 flex-col p-4 pt-0`,children:M(`div`,{className:`flex flex-col items-center gap-y-1`,children:[j(`p`,{className:`text-sm text-text-positive-weak`,children:`Đăng nhập với Google`}),j(U,{size:`large`,theme:`outline`,onSuccess:async e=>{!e.clientId||!e.credential||!e.select_by||await i?.(e)}})]})})}),j(w,{})]})})]})},Le=({children:e,...t})=>{let{isLoggedIn:n}=G();return n?j(Te,{variant:`inset`,collapsible:`icon`,...t,children:e}):null},Re=({children:e})=>{let{isLoggedIn:t,onGoogleLoginSuccess:r}=G(),i=H(`(min-width: 640px)`),[a,o]=B(!1);return t?j(Ee,{children:j(`section`,{className:`relative size-full`,children:j(`div`,{className:`absolute inset-0 flex flex-col`,children:e})})}):j(`div`,{className:`size-full p-4 pt-[calc(var(--header-height)+1.5rem)]`,children:M(`div`,{className:`flex size-full flex-col items-center justify-center gap-6 rounded-lg bg-card p-8 text-center shadow-card`,children:[j(`div`,{className:`flex size-20 items-center justify-center rounded-full bg-muted-foreground/10`,children:j(de,{className:`size-10 text-primary`})}),M(`div`,{className:`flex max-w-md flex-col gap-2`,children:[j(`h2`,{className:`font-semibold text-2xl`,children:`Bạn chưa đăng nhập`}),j(`p`,{className:`text-text-positive-weak`,children:`Đăng nhập để khám phá đầy đủ các tính năng của Lunas Store và truy cập vào tài khoản của bạn.`})]}),M(n,{size:`lg`,className:`gap-2`,onClick:()=>o(!0),children:[j(N,{size:18}),j(`span`,{children:`Đăng nhập ngay`})]}),i&&j(h,{open:a,onOpenChange:o,children:M(m,{showCloseButton:!1,className:`flex flex-col gap-0 border-none p-0 sm:max-w-sm`,children:[j(p,{className:`flex-0 gap-2 p-6`,children:j(f,{className:`text-center`,children:`Chào mừng bạn đến với Lunas Store!`})}),j(`div`,{className:`flex flex-1 flex-col`,children:j(`main`,{className:`size-full flex-1 bg-card p-4 pt-0`,children:M(`div`,{className:`flex flex-col items-center gap-y-1`,children:[j(`p`,{className:`text-sm text-text-positive-weak`,children:`Đăng nhập với Google`}),j(U,{size:`large`,theme:`outline`,width:240,onSuccess:async e=>{!e.clientId||!e.credential||!e.select_by||(await r?.(e),o(!1))}})]})})}),j(d,{className:`p-2`})]})}),!i&&j(D,{open:a,onOpenChange:o,children:M(E,{children:[j(oe,{className:`text-left`,children:j(T,{children:`Chào mừng bạn đến với Lunas Store!`})}),j(`div`,{className:`flex flex-1 flex-col`,children:j(`main`,{className:`flex size-full flex-1 flex-col p-4 pt-0`,children:M(`div`,{className:`flex flex-col items-center gap-y-1`,children:[j(`p`,{className:`text-sm text-text-positive-weak`,children:`Đăng nhập với Google`}),j(U,{size:`large`,theme:`outline`,onSuccess:async e=>{!e.clientId||!e.credential||!e.select_by||(await r?.(e),o(!1))}})]})})}),j(w,{})]})})]})})},ze=({className:e,children:t})=>j(`div`,{"data-slot":`main-header`,className:A(`flex-0 snap-start`,e),children:t}),Be=({className:e,children:t})=>j(`div`,{"data-slot":`main-content`,className:A(`flex w-full flex-1 flex-col gap-4 overflow-y-auto px-2 sm:px-4`,e),children:t}),Ve=({className:e,children:t})=>j(`div`,{"data-slot":`main-footer`,className:A(`hidden w-full flex-0 border-border-weak border-t pt-2 sm:flex`,e),children:t}),He=({children:e,className:t})=>j(`div`,{"data-slot":`main-group`,className:A(`flex size-full flex-col gap-4`,t),children:e}),Ue=({className:e,children:t})=>j(`div`,{"data-slot":`main-group-content`,className:A(`size-full max-w-8xl flex-1 rounded-md bg-card p-4 shadow-card`,e),children:t});export{Fe as ServiceLayoutCartInfo,Ie as ServiceLayoutHeader,Re as ServiceLayoutMain,Be as ServiceLayoutMainContent,Ve as ServiceLayoutMainFooter,He as ServiceLayoutMainGroup,Ue as ServiceLayoutMainGroupContent,ze as ServiceLayoutMainHeader,Ne as ServiceLayoutProvider,Le as ServiceLayoutSidebar,Oe as ServiceLayoutSidebarContent,De as ServiceLayoutSidebarFooter,ke as ServiceLayoutSidebarGroup,je as ServiceLayoutSidebarGroupContent,Ae as ServiceLayoutSidebarGroupLabel,X as ServiceLayoutSidebarMenu,Q as ServiceLayoutSidebarMenuButton,Z as ServiceLayoutSidebarMenuItem,$ as ServiceLayoutUserInfo,Pe as ServiceLayoutWrapper};
|
|
2
2
|
//# sourceMappingURL=index.mjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.mjs","names":["CartItem: React.FC<\n CartItemProps & {\n type: 'in_stock' | 'pre_order';\n }\n>","Image","CartList: React.FC<CartListProps>","ScrollArea","Separator","open","ServiceLayoutSidebar","SlotPrimitive","Tooltip","ServiceLayoutProvider: React.FC<React.PropsWithChildren<ServiceLayoutContextProps>>","ServiceLayoutWrapper: React.FC<React.PropsWithChildren>","ServiceLayoutUserInfo: React.FC<ServiceLayoutUserInfoProps>","DropdownMenu","Avatar","ServiceLayoutCartInfo: React.FC","Tabs","ServiceLayoutHeader: React.FC<ServiceLayoutHeaderProps>","Separator","Dialog","ServiceLayoutSidebar: React.FC<React.PropsWithChildren & React.ComponentProps<typeof LayoutSidebar>>","LayoutSidebar","ServiceLayoutMain: React.FC<React.PropsWithChildren>","ServiceLayoutMainHeader: React.FC<React.PropsWithChildren & { className?: string }>","ServiceLayoutMainContent: React.FC<React.PropsWithChildren & { className?: string }>","ServiceLayoutMainFooter: React.FC<React.PropsWithChildren & { className?: string }>","ServiceLayoutMainGroup: React.FC<React.PropsWithChildren<{ className?: string }>>","ServiceLayoutMainGroupContent: React.FC<React.PropsWithChildren & { className?: string }>"],"sources":["../../../packages/components/layouts/service-layout/hooks/use-service-layout.ts","../../../packages/components/layouts/service-layout/components/cart/index.tsx","../../../packages/components/layouts/service-layout/hooks/use-service-layout-sidebar.ts","../../../packages/components/layouts/service-layout/service-layout-sidebar.tsx","../../../packages/components/layouts/service-layout/service-layout.tsx"],"sourcesContent":["import { createContext, use } from 'react';\n\nimport type { CredentialResponse } from '@react-oauth/google';\n\ntype Cart = {\n id: string;\n productUuid: string;\n productName: string;\n variantUuid: string;\n variantName: string;\n imageUrl: string;\n options: Array<{ label: string; value: string }>;\n quantity: number;\n price: number;\n};\n\nexport type ServiceLayoutContextProps = {\n isLoggedIn?: boolean;\n username?: string;\n email?: string;\n inStockCarts?: Cart[];\n orderedCarts?: Cart[];\n onGoogleLoginSuccess?: (params: CredentialResponse) => void | Promise<void>;\n onUpdatingCart?: (id: string, quantity: number, type: 'in_stock' | 'pre_order') => void | Promise<void>;\n onDeletingCart?: (id: string) => void | Promise<void>;\n onLogout?: () => void | Promise<void>;\n};\n\nexport const ServiceLayoutContext = createContext<ServiceLayoutContextProps | null>(null);\n\nexport const useServiceLayout = () => {\n const context = use(ServiceLayoutContext);\n if (!context) {\n throw new Error('useServiceLayoutContext must be used within a ServiceLayoutProvider');\n }\n return context;\n};\n","import { useCallback, useEffect, useState } from 'react';\n\nimport { Minus, Plus, ShoppingBasketIcon, Trash2 } from 'lucide-react';\n\nimport { useDebounceCallback } from '@customafk/react-toolkit/hooks/useDebounceCallback';\nimport { cn } from '@customafk/react-toolkit/utils';\n\nimport { Button } from '@/components/ui/button';\nimport { Card, CardContent } from '@/components/ui/card';\nimport { Image } from '@/components/ui/image';\nimport { Input } from '@/components/ui/input';\nimport { ScrollArea } from '@/components/ui/scroll-area';\nimport { Separator } from '@/components/ui/separator';\nimport { useServiceLayout } from '../../hooks/use-service-layout';\n\ntype CartItemProps = {\n id: string;\n productUuid: string;\n productName: string;\n variantUuid: string;\n variantName: string;\n imageUrl: string;\n options: Array<{ label: string; value: string }>;\n quantity: number;\n price: number;\n};\n\nexport const CartItem: React.FC<\n CartItemProps & {\n type: 'in_stock' | 'pre_order';\n }\n> = ({ id, type, productName, imageUrl, quantity, price, options }) => {\n const { onDeletingCart, onUpdatingCart } = useServiceLayout();\n const [itemQuantity, setItemQuantity] = useState(quantity);\n const [isDeleting, setIsDeleting] = useState<boolean>(false);\n\n const handleUpdate = useCallback(() => {\n onUpdatingCart?.(id, itemQuantity, type);\n }, [onUpdatingCart, id, itemQuantity, type]);\n\n const handleUpdateDebounced = useDebounceCallback(handleUpdate, 500);\n\n const handleQuantityChange = useCallback((value: number) => {\n if (value < 1) value = 1;\n if (value > 99) value = 99;\n setItemQuantity(value);\n }, []);\n\n const handleRemoveItem = useCallback(async () => {\n setIsDeleting(true);\n await onDeletingCart?.(id);\n setIsDeleting(false);\n }, [id, onDeletingCart]);\n\n useEffect(() => {\n if (itemQuantity !== quantity) {\n handleUpdateDebounced();\n }\n return () => {\n handleUpdateDebounced.cancel();\n };\n }, [itemQuantity, quantity, handleUpdateDebounced]);\n\n return (\n <Card className=\"border-border-weak relative mb-3 overflow-x-auto border p-4 shadow-none\">\n {isDeleting && (\n <div className=\"bg-muted-muted/80 absolute inset-0 z-10 flex items-center justify-center\">\n <div className=\"loader-dots\" />\n </div>\n )}\n <CardContent className=\"p-0\">\n <Button\n variant=\"ghost\"\n size=\"icon\"\n color=\"muted\"\n disabled={isDeleting}\n className=\"text-text-positive-weak absolute top-2 right-2 z-10\"\n onClick={handleRemoveItem}\n >\n <Trash2 className=\"h-4 w-4\" />\n <span className=\"sr-only\">Remove item</span>\n </Button>\n <div className=\"flex gap-3\">\n <div className=\"relative size-20 flex-shrink-0\">\n {imageUrl ? (\n <Image src={imageUrl} alt={productName} className=\"rounded-md border-none shadow-xs\" width={80} height={80} />\n ) : (\n <div className=\"bg-muted text-muted-foreground flex size-full items-center justify-center text-xs\">No image</div>\n )}\n </div>\n\n <div className=\"flex flex-1 flex-col\">\n <div className=\"flex justify-between\">\n <h3 className=\"text-text-positive line-clamp-1 text-sm font-medium\">{productName}</h3>\n </div>\n\n <p className=\"text-text-positive-weak mb-1 text-xs\">\n {options.map((option, index) => (\n <span key={option.label}>\n {option.label}: {option.value}\n {index < options.length - 1 && ', '}\n </span>\n ))}\n </p>\n\n <div className=\"mt-auto flex items-center justify-between\">\n <div className=\"flex items-center space-x-1\">\n <button\n disabled={itemQuantity <= 1}\n className=\"border-border active:bg-muted-muted flex size-6 cursor-pointer items-center justify-center rounded-full border transition-all disabled:opacity-60\"\n onClick={() => handleQuantityChange(itemQuantity - 1)}\n >\n <Minus size={12} />\n </button>\n <Input\n value={itemQuantity}\n onChange={e => handleQuantityChange(parseInt(e.target.value || '1'))}\n className=\"border-border h-6.5 w-14 rounded-md border p-1 text-center text-sm\"\n min={1}\n />\n <button\n disabled={itemQuantity >= 99}\n className=\"border-border active:bg-muted-muted flex size-6 cursor-pointer items-center justify-center rounded-full border transition-all disabled:opacity-60\"\n onClick={() => handleQuantityChange(itemQuantity + 1)}\n >\n <Plus size={12} />\n </button>\n </div>\n\n <div className=\"text-sm font-semibold\">{(price * itemQuantity).toLocaleString()} ₫</div>\n </div>\n </div>\n </div>\n </CardContent>\n </Card>\n );\n};\n\ntype CartListProps = {\n items?: CartItemProps[];\n cartType: 'in_stock' | 'pre_order';\n className?: string;\n};\n\nexport const CartList: React.FC<CartListProps> = ({ items = [], cartType, className }) => {\n const totalAmount = items.reduce((sum, item) => sum + item.price * item.quantity, 0);\n\n if (items.length === 0) {\n return (\n <div className={cn('bg-muted-muted flex size-full max-h-80 flex-col items-center justify-center rounded-lg p-8', className)}>\n <div className=\"text-text-positive-weak bg-card shadow-card mb-4 flex size-20 items-center justify-center rounded-full text-5xl\">\n <ShoppingBasketIcon size={42} strokeWidth={1} />\n </div>\n <div className=\"flex flex-col space-y-1\">\n <p className=\"text-text-positive text-center text-base font-semibold\">Giỏ hàng trống</p>\n <p className=\"text-text-positive-weak mb-4 text-center text-sm\">\n {cartType === 'in_stock' ? 'Thêm sản phẩm có sẵn vào giỏ hàng của bạn!' : 'Thêm sản phẩm đặt trước vào giỏ hàng của bạn!'}\n </p>\n </div>\n </div>\n );\n }\n\n return (\n <div className={cn('flex h-full flex-col space-y-4 overflow-y-auto', className)}>\n <ScrollArea className=\"h-full flex-1\">\n {items.map(item => (\n <CartItem key={item.id} {...item} type={cartType} />\n ))}\n </ScrollArea>\n\n <div className=\"flex-0 space-y-3\">\n <div className=\"flex justify-between\">\n <span className=\"text-text-positive-weak text-sm\">Tạm tính:</span>\n <span className=\"text-sm\">{totalAmount.toLocaleString()} ₫</span>\n </div>\n <Separator />\n <div className=\"flex justify-between\">\n <span className=\"text-text-positive font-medium\">Tổng cộng:</span>\n <span className=\"text-lg font-semibold\">{totalAmount.toLocaleString()} ₫</span>\n </div>\n <Button className=\"mt-2 w-full\">Thanh toán ngay</Button>\n {cartType === 'pre_order' && <p className=\"text-text-positive-weak text-center text-xs italic\">* Sản phẩm đặt trước sẽ được giao sau khi có hàng</p>}\n </div>\n </div>\n );\n};\n","import { createContext, use } from 'react';\n\nexport type SidebarContextProps = {\n state: 'expanded' | 'collapsed';\n open: boolean;\n setOpen: (open: boolean) => void;\n openMobile: boolean;\n setOpenMobile: (open: boolean) => void;\n isMobile: boolean;\n toggleSidebar: () => void;\n};\n\nexport const SidebarContext = createContext<SidebarContextProps | null>(null);\n\nexport const useServiceLayoutSidebar = () => {\n const context = use(SidebarContext);\n if (!context) {\n throw new Error('useSidebar must be used within a SidebarProvider.');\n }\n\n return context;\n};\n","import { useCallback, useEffect, useMemo, useState } from 'react';\n\nimport { LogOutIcon, MenuIcon, ShoppingCartIcon } from 'lucide-react';\n\nimport { useIsMobile } from '@customafk/react-toolkit/hooks/useMobile';\nimport { cn } from '@customafk/react-toolkit/utils';\n\nimport { Button } from '@/components/ui/button';\nimport { Separator } from '@/components/ui/separator';\nimport { Sheet, SheetContent, SheetDescription, SheetHeader, SheetTitle } from '@/components/ui/sheet';\nimport { Skeleton } from '@/components/ui/skeleton';\nimport { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';\n\nimport { cva, type VariantProps } from 'class-variance-authority';\nimport { Slot as SlotPrimitive } from 'radix-ui';\nimport { useServiceLayout } from './hooks/use-service-layout';\nimport { SidebarContext, type SidebarContextProps, useServiceLayoutSidebar } from './hooks/use-service-layout-sidebar';\n\nconst SIDEBAR_COOKIE_NAME = 'sidebar_state';\nconst SIDEBAR_COOKIE_MAX_AGE = 60 * 60 * 24 * 7;\nconst SIDEBAR_WIDTH = '16rem';\nconst SIDEBAR_WIDTH_MOBILE = '16rem';\nconst SIDEBAR_WIDTH_ICON = '3rem';\nconst SIDEBAR_KEYBOARD_SHORTCUT = 'b';\n\nfunction ServiceLayoutSidebarProvider({\n defaultOpen = true,\n open: openProp,\n onOpenChange: setOpenProp,\n className,\n style,\n children,\n ...props\n}: React.ComponentProps<'div'> & {\n defaultOpen?: boolean;\n open?: boolean;\n onOpenChange?: (open: boolean) => void;\n}) {\n const isMobile = useIsMobile();\n const [openMobile, setOpenMobile] = useState(false);\n\n // This is the internal state of the sidebar.\n // We use openProp and setOpenProp for control from outside the component.\n const [_open, _setOpen] = useState(defaultOpen);\n const open = openProp ?? _open;\n const setOpen = useCallback(\n (value: boolean | ((value: boolean) => boolean)) => {\n const openState = typeof value === 'function' ? value(open) : value;\n if (setOpenProp) {\n setOpenProp(openState);\n } else {\n _setOpen(openState);\n }\n\n // This sets the cookie to keep the sidebar state.\n // biome-ignore lint/suspicious/noDocumentCookie: all\n document.cookie = `${SIDEBAR_COOKIE_NAME}=${openState}; path=/; max-age=${SIDEBAR_COOKIE_MAX_AGE}`;\n },\n [setOpenProp, open]\n );\n\n // Helper to toggle the sidebar.\n const toggleSidebar = useCallback(() => {\n return isMobile ? setOpenMobile(open => !open) : setOpen(open => !open);\n }, [isMobile, setOpen]);\n\n // Adds a keyboard shortcut to toggle the sidebar.\n useEffect(() => {\n const handleKeyDown = (event: KeyboardEvent) => {\n if (event.key === SIDEBAR_KEYBOARD_SHORTCUT && (event.metaKey || event.ctrlKey)) {\n event.preventDefault();\n toggleSidebar();\n }\n };\n\n window.addEventListener('keydown', handleKeyDown);\n return () => window.removeEventListener('keydown', handleKeyDown);\n }, [toggleSidebar]);\n\n // We add a state so that we can do data-state=\"expanded\" or \"collapsed\".\n // This makes it easier to style the sidebar with Tailwind classes.\n const state = open ? 'expanded' : 'collapsed';\n\n const contextValue = useMemo<SidebarContextProps>(\n () => ({\n state,\n isMobile,\n\n toggleSidebar,\n\n open,\n setOpen,\n\n openMobile,\n setOpenMobile,\n }),\n [state, open, setOpen, isMobile, openMobile, toggleSidebar]\n );\n\n return (\n <SidebarContext.Provider value={contextValue}>\n <TooltipProvider delayDuration={0}>\n <div\n data-slot=\"sidebar-wrapper\"\n style={\n {\n '--sidebar-width': SIDEBAR_WIDTH,\n '--sidebar-width-icon': SIDEBAR_WIDTH_ICON,\n ...style,\n } as React.CSSProperties\n }\n className={cn('group/sidebar-wrapper', 'has-data-[variant=inset]:bg-sidebar', 'flex h-dvh w-full', className)}\n {...props}\n >\n {children}\n </div>\n </TooltipProvider>\n </SidebarContext.Provider>\n );\n}\n\nfunction ServiceLayoutSidebar({\n side = 'left',\n variant = 'sidebar',\n collapsible = 'offcanvas',\n className,\n children,\n ...props\n}: React.ComponentProps<'div'> & {\n side?: 'left' | 'right';\n variant?: 'sidebar' | 'floating' | 'inset';\n collapsible?: 'offcanvas' | 'icon' | 'none';\n}) {\n const { isMobile, state, openMobile, setOpenMobile } = useServiceLayoutSidebar();\n\n if (collapsible === 'none') {\n return (\n <aside data-slot=\"sidebar\" className={cn('flex h-full w-(--sidebar-width) flex-col bg-sidebar text-sidebar-foreground', className)} {...props}>\n {children}\n </aside>\n );\n }\n\n if (isMobile) {\n return (\n <Sheet open={openMobile} onOpenChange={setOpenMobile}>\n <SheetContent\n side=\"left\"\n data-sidebar=\"sidebar\"\n data-slot=\"sidebar\"\n data-mobile=\"true\"\n className=\"w-(--sidebar-width) bg-sidebar p-0 text-sidebar-foreground sm:max-w-3xs [&>button]:hidden\"\n style={\n {\n '--sidebar-width': SIDEBAR_WIDTH_MOBILE,\n } as React.CSSProperties\n }\n >\n <SheetHeader className=\"sr-only\">\n <SheetTitle>Sidebar</SheetTitle>\n <SheetDescription>Displays the mobile sidebar.</SheetDescription>\n </SheetHeader>\n <div className=\"flex size-full flex-col\">\n <div className=\"flex flex-0 items-center gap-x-2 border-border-weak border-b p-2 pr-4\">\n <ServiceLayoutSidebarTrigger />\n <div className=\"ml-2 flex aspect-square size-8 items-center justify-center rounded-lg bg-sidebar-primary text-sidebar-primary-foreground\">\n <ShoppingCartIcon size={20} />\n </div>\n <div className=\"grid flex-1 text-left text-sm leading-tight\">\n <span className=\"truncate font-medium\">Lunas Store</span>\n <span className=\"truncate text-xs\">Established 2023</span>\n </div>\n </div>\n <div className=\"flex flex-1 flex-col p-2\">{children}</div>\n </div>\n </SheetContent>\n </Sheet>\n );\n }\n\n return (\n <aside\n className=\"group peer hidden bg-card text-sidebar-foreground md:block\"\n data-state={state}\n data-collapsible={state === 'collapsed' ? collapsible : ''}\n data-variant={variant}\n data-side={side}\n data-slot=\"sidebar\"\n >\n {/* This is what handles the sidebar gap on desktop */}\n <div\n data-slot=\"sidebar-gap\"\n className={cn(\n 'relative',\n 'bg-transparent',\n 'transition-[width] duration-200 ease-linear',\n 'h-(--header-height) w-(--sidebar-width)',\n 'sm:h-[calc(var(--header-height)+0.5rem)]',\n 'group-data-[collapsible=offcanvas]:w-0',\n 'group-data-[side=right]:rotate-180',\n variant === 'floating' || variant === 'inset'\n ? 'group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4)))]'\n : 'group-data-[collapsible=icon]:w-(--sidebar-width-icon)'\n )}\n />\n <div\n data-slot=\"sidebar-container\"\n className={cn(\n 'hidden md:flex',\n 'fixed inset-y-0 top-14 z-10 shadow-nav',\n 'h-[calc(100dvh-3.5rem)] w-(--sidebar-width)',\n 'transition-[left,right,width] duration-200 ease-linear',\n side === 'left' && 'left-0 group-data-[collapsible=offcanvas]:left-[calc(var(--sidebar-width)*-1)]',\n side === 'right' && 'right-0 group-data-[collapsible=offcanvas]:right-[calc(var(--sidebar-width)*-1)]',\n // Adjust the padding for floating and inset variants.\n variant === 'floating' || variant === 'inset'\n ? 'p-2 group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4)))]'\n : 'group-data-[collapsible=icon]:w-(--sidebar-width-icon) group-data-[side=left]:border-r group-data-[side=right]:border-l',\n className\n )}\n {...props}\n >\n <div\n data-sidebar=\"sidebar\"\n data-slot=\"sidebar-inner\"\n className={cn(\n 'flex size-full flex-col',\n 'group-data-[variant=floating]:rounded-lg',\n 'group-data-[variant=floating]:border',\n 'group-data-[variant=floating]:border-sidebar-border',\n 'group-data-[variant=floating]:shadow-sm'\n )}\n >\n {children}\n </div>\n </div>\n </aside>\n );\n}\n\nfunction ServiceLayoutSidebarTrigger({ className, onClick, ...props }: React.ComponentProps<typeof Button>) {\n const { toggleSidebar } = useServiceLayoutSidebar();\n\n return (\n <Button\n data-sidebar=\"trigger\"\n data-slot=\"sidebar-trigger\"\n variant=\"ghost\"\n color=\"muted\"\n size=\"icon\"\n className={cn('size-10 rounded-full', className)}\n onClick={event => {\n onClick?.(event);\n toggleSidebar();\n }}\n {...props}\n >\n <MenuIcon className=\"size-6!\" />\n <span className=\"sr-only\">Toggle Sidebar</span>\n </Button>\n );\n}\n\nfunction ServiceLayoutSidebarRail({ className, ...props }: React.ComponentProps<'button'>) {\n const { toggleSidebar } = useServiceLayoutSidebar();\n\n return (\n <button\n data-sidebar=\"rail\"\n data-slot=\"sidebar-rail\"\n aria-label=\"Toggle Sidebar\"\n tabIndex={-1}\n onClick={toggleSidebar}\n title=\"Toggle Sidebar\"\n className={cn(\n 'absolute inset-y-0 z-20 hidden w-4 -translate-x-1/2 transition-all ease-linear sm:flex',\n 'after:absolute after:inset-y-0 after:left-1/2 after:w-0.5',\n 'group-data-[side=left]:-right-4',\n 'group-data-[side=right]:left-0',\n 'in-data-[side=left]:cursor-w-resize',\n 'in-data-[side=right]:cursor-e-resize',\n 'hover:after:bg-sidebar-border',\n 'hover:group-data-[collapsible=offcanvas]:bg-sidebar',\n 'group-data-[collapsible=offcanvas]:translate-x-0',\n 'group-data-[collapsible=offcanvas]:after:left-full',\n '[[data-side=left][data-state=collapsed]_&]:cursor-e-resize',\n '[[data-side=right][data-state=collapsed]_&]:cursor-w-resize',\n '[[data-side=left][data-collapsible=offcanvas]_&]:-right-2',\n '[[data-side=right][data-collapsible=offcanvas]_&]:-left-2',\n className\n )}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarInset({ className, children, ...props }: React.ComponentProps<'main'>) {\n return (\n <main data-slot=\"sidebar-inset\" className={cn('relative flex w-full flex-1 flex-col', className)} {...props}>\n <div className=\"h-(--header-height) w-full sm:h-[calc(var(--header-height)+0.5rem)]\" />\n <div className={cn('inset-shadow-sm flex-1')}>{children}</div>\n </main>\n );\n}\n\nfunction ServiceLayoutSidebarHeader({ className, ...props }: React.ComponentProps<'div'>) {\n return <div data-slot=\"sidebar-header\" data-sidebar=\"header\" className={cn('flex flex-col gap-2 p-2', className)} {...props} />;\n}\n\nfunction ServiceLayoutSidebarFooter({ className, children, ...props }: React.ComponentProps<'div'>) {\n const { open } = useServiceLayoutSidebar();\n const { onLogout } = useServiceLayout();\n return (\n <div data-slot=\"sidebar-footer\" data-sidebar=\"footer\" className={cn('flex flex-col gap-2', className)} {...props}>\n {children}\n <ServiceLayoutSidebarMenu>\n <ServiceLayoutSidebarMenuItem>\n <ServiceLayoutSidebarMenuButton className=\"border border-border\" onClick={onLogout}>\n <LogOutIcon className=\"text-text-positive-weak\" />\n Đăng xuất\n </ServiceLayoutSidebarMenuButton>\n </ServiceLayoutSidebarMenuItem>\n {open && (\n <ServiceLayoutSidebarMenuItem className=\"mt-2 border-t border-t-border\">\n <p className=\"pt-2 text-center text-muted-foreground text-xs\">Copyright © 2025, Lunas.</p>\n </ServiceLayoutSidebarMenuItem>\n )}\n </ServiceLayoutSidebarMenu>\n </div>\n );\n}\n\nfunction ServiceLayoutSidebarSeparator({ className, ...props }: React.ComponentProps<typeof Separator>) {\n return <Separator data-slot=\"sidebar-separator\" data-sidebar=\"separator\" className={cn('mx-2 w-auto bg-sidebar-border', className)} {...props} />;\n}\n\nfunction ServiceLayoutSidebarContent({ className, ...props }: React.ComponentProps<'div'>) {\n return (\n <div\n data-slot=\"sidebar-content\"\n data-sidebar=\"content\"\n className={cn('flex min-h-0 flex-1 flex-col gap-2 overflow-auto group-data-[collapsible=icon]:overflow-hidden', className)}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarGroup({ className, ...props }: React.ComponentProps<'div'>) {\n return <div data-slot=\"sidebar-group\" data-sidebar=\"group\" className={cn('relative flex w-full min-w-0 flex-col', className)} {...props} />;\n}\n\nfunction ServiceLayoutSidebarGroupLabel({ className, asChild = false, ...props }: React.ComponentProps<'div'> & { asChild?: boolean }) {\n const Comp = asChild ? SlotPrimitive.Slot : 'div';\n\n return (\n <Comp\n data-slot=\"sidebar-group-label\"\n data-sidebar=\"group-label\"\n className={cn(\n 'flex h-8 shrink-0 items-center rounded-md px-2 font-medium text-xs outline-hidden transition-[margin,opacity] duration-200 ease-linear focus-visible:ring-2',\n 'text-sidebar-foreground/70',\n 'ring-sidebar-ring',\n '[&>svg]:size-4',\n '[&>svg]:shrink-0',\n 'group-data-[collapsible=icon]:-mt-8',\n 'group-data-[collapsible=icon]:opacity-0',\n className\n )}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarGroupAction({ className, asChild = false, ...props }: React.ComponentProps<'button'> & { asChild?: boolean }) {\n const Comp = asChild ? SlotPrimitive.Slot : 'button';\n\n return (\n <Comp\n data-slot=\"sidebar-group-action\"\n data-sidebar=\"group-action\"\n className={cn(\n 'absolute top-3.5 right-3 flex aspect-square w-5 items-center justify-center rounded-md p-0 text-sidebar-foreground outline-hidden ring-sidebar-ring transition-transform hover:bg-sidebar-accent hover:text-sidebar-accent-foreground',\n 'focus-visible:ring-2',\n '[&>svg]:size-4',\n '[&>svg]:shrink-0',\n // Increases the hit area of the button on mobile.\n 'after:absolute after:-inset-2 md:after:hidden',\n 'group-data-[collapsible=icon]:hidden',\n className\n )}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarGroupContent({ className, ...props }: React.ComponentProps<'div'>) {\n return <div data-slot=\"sidebar-group-content\" data-sidebar=\"group-content\" className={cn('w-full text-sm', className)} {...props} />;\n}\n\nfunction ServiceLayoutSidebarMenu({ className, ...props }: React.ComponentProps<'ul'>) {\n return <ul data-slot=\"sidebar-menu\" data-sidebar=\"menu\" className={cn('flex w-full min-w-0 flex-col gap-1', className)} {...props} />;\n}\n\nfunction ServiceLayoutSidebarMenuItem({ className, ...props }: React.ComponentProps<'li'>) {\n return <li data-slot=\"sidebar-menu-item\" data-sidebar=\"menu-item\" className={cn('group/menu-item relative', className)} {...props} />;\n}\n\nconst sidebarMenuButtonVariants = cva(\n [\n 'peer/menu-button',\n 'cursor-pointer',\n 'flex w-full items-center gap-2',\n 'overflow-hidden rounded-md p-2 outline-hidden',\n 'truncate text-left',\n 'transition-[color,width,height,padding]',\n 'hover:bg-sidebar-accent',\n 'hover:text-sidebar-accent-foreground',\n 'active:bg-sidebar-accent',\n 'active:text-sidebar-accent-foreground',\n 'disabled:pointer-events-none',\n 'disabled:opacity-50',\n 'group-has-data-[sidebar=menu-action]/menu-item:pr-8',\n 'aria-disabled:pointer-events-none',\n 'aria-disabled:opacity-50',\n 'data-[active=true]:bg-sidebar-primary-muted',\n 'data-[active=true]:font-medium',\n 'data-[active=true]:text-sidebar-primary',\n 'data-[state=open]:hover:bg-sidebar-accent',\n 'data-[state=open]:hover:text-sidebar-accent-foreground',\n 'group-data-[collapsible=icon]:size-12!',\n 'group-data-[collapsible=icon]:p-3!',\n 'group-data-[collapsible=icon]:gap-3!',\n '[&>svg]:size-6',\n '[&>svg]:shrink-0',\n '[&>span:last-child]:truncate',\n ],\n {\n variants: {\n variant: {\n default: 'hover:bg-sidebar-accent/60 hover:text-sidebar-accent-foreground/80',\n outline:\n 'bg-background shadow-[0_0_0_1px_hsl(var(--sidebar-border))] hover:bg-sidebar-accent hover:text-sidebar-accent-foreground hover:shadow-[0_0_0_1px_hsl(var(--sidebar-accent))]',\n },\n size: {\n default: 'h-10 text-sm',\n sm: 'h-7 text-xs',\n lg: 'h-12 text-sm group-data-[collapsible=icon]:p-0!',\n },\n },\n defaultVariants: {\n variant: 'default',\n size: 'default',\n },\n }\n);\n\nfunction ServiceLayoutSidebarMenuButton({\n asChild = false,\n isActive = false,\n variant = 'default',\n size = 'default',\n tooltip,\n className,\n ...props\n}: React.ComponentProps<'button'> & {\n asChild?: boolean;\n isActive?: boolean;\n tooltip?: string | React.ComponentProps<typeof TooltipContent>;\n} & VariantProps<typeof sidebarMenuButtonVariants>) {\n const Comp = asChild ? SlotPrimitive.Slot : 'button';\n const { isMobile, state } = useServiceLayoutSidebar();\n\n const button = (\n <Comp\n data-slot=\"sidebar-menu-button\"\n data-sidebar=\"menu-button\"\n data-size={size}\n data-active={isActive}\n className={cn(sidebarMenuButtonVariants({ variant, size }), className)}\n {...props}\n />\n );\n\n if (!tooltip) {\n return button;\n }\n\n if (typeof tooltip === 'string') {\n tooltip = {\n children: tooltip,\n };\n }\n\n return (\n <Tooltip>\n <TooltipTrigger asChild>{button}</TooltipTrigger>\n <TooltipContent side=\"right\" align=\"center\" hidden={state !== 'collapsed' || isMobile} {...tooltip} />\n </Tooltip>\n );\n}\n\nfunction ServiceLayoutSidebarMenuAction({\n className,\n asChild = false,\n showOnHover = false,\n ...props\n}: React.ComponentProps<'button'> & {\n asChild?: boolean;\n showOnHover?: boolean;\n}) {\n const Comp = asChild ? SlotPrimitive.Slot : 'button';\n\n return (\n <Comp\n data-slot=\"sidebar-menu-action\"\n data-sidebar=\"menu-action\"\n className={cn(\n 'absolute top-1.5 right-1 flex aspect-square w-5 items-center justify-center rounded-md p-0 text-sidebar-foreground outline-hidden ring-sidebar-ring transition-transform hover:bg-sidebar-accent hover:text-sidebar-accent-foreground peer-hover/menu-button:text-sidebar-accent-foreground',\n 'focus-visible:ring-2',\n '[&>svg]:size-4',\n '[&>svg]:shrink-0',\n // Increases the hit area of the button on mobile.\n 'after:absolute after:-inset-2 md:after:hidden',\n 'peer-data-[size=sm]/menu-button:top-1',\n 'peer-data-[size=default]/menu-button:top-1.5',\n 'peer-data-[size=lg]/menu-button:top-2.5',\n 'group-data-[collapsible=icon]:hidden',\n showOnHover && 'peer-data-[active=true]/menu-button:text-sidebar-accent-foreground',\n showOnHover && 'group-focus-within/menu-item:opacity-100',\n showOnHover && 'group-hover/menu-item:opacity-100',\n showOnHover && 'data-[state=open]:opacity-100 md:opacity-0',\n className\n )}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarMenuBadge({ className, ...props }: React.ComponentProps<'div'>) {\n return (\n <div\n data-slot=\"sidebar-menu-badge\"\n data-sidebar=\"menu-badge\"\n className={cn(\n 'pointer-events-none absolute right-1 flex h-5 min-w-5 select-none items-center justify-center rounded-md px-1 font-medium text-sidebar-foreground text-xs tabular-nums',\n 'peer-hover/menu-button:text-sidebar-accent-foreground',\n 'peer-data-[active=true]/menu-button:text-sidebar-accent-foreground',\n 'peer-data-[size=sm]/menu-button:top-1',\n 'peer-data-[size=default]/menu-button:top-1.5',\n 'peer-data-[size=lg]/menu-button:top-2.5',\n 'group-data-[collapsible=icon]:hidden',\n className\n )}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarMenuSkeleton({\n className,\n showIcon = false,\n ...props\n}: React.ComponentProps<'div'> & {\n showIcon?: boolean;\n}) {\n // Random width between 50 to 90%.\n const width = useMemo(() => {\n return `${Math.floor(Math.random() * 40) + 50}%`;\n }, []);\n\n return (\n <div data-slot=\"sidebar-menu-skeleton\" data-sidebar=\"menu-skeleton\" className={cn('flex h-8 items-center gap-2 rounded-md px-2', className)} {...props}>\n {showIcon && <Skeleton className=\"size-4 rounded-md\" data-sidebar=\"menu-skeleton-icon\" />}\n <Skeleton\n className=\"h-4 max-w-(--skeleton-width) flex-1\"\n data-sidebar=\"menu-skeleton-text\"\n style={\n {\n '--skeleton-width': width,\n } as React.CSSProperties\n }\n />\n </div>\n );\n}\n\nfunction ServiceLayoutSidebarMenuSub({ className, ...props }: React.ComponentProps<'ul'>) {\n return (\n <ul\n data-slot=\"sidebar-menu-sub\"\n data-sidebar=\"menu-sub\"\n className={cn(\n 'mx-3.5 flex min-w-0 translate-x-px flex-col gap-1 border-sidebar-border border-l px-2.5 py-0.5',\n 'group-data-[collapsible=icon]:hidden',\n className\n )}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarMenuSubItem({ className, ...props }: React.ComponentProps<'li'>) {\n return <li data-slot=\"sidebar-menu-sub-item\" data-sidebar=\"menu-sub-item\" className={cn('group/menu-sub-item relative', className)} {...props} />;\n}\n\nfunction ServiceLayoutSidebarMenuSubButton({\n asChild = false,\n size = 'md',\n isActive = false,\n className,\n ...props\n}: React.ComponentProps<'a'> & {\n asChild?: boolean;\n size?: 'sm' | 'md';\n isActive?: boolean;\n}) {\n const Comp = asChild ? SlotPrimitive.Slot : 'a';\n\n return (\n <Comp\n data-slot=\"sidebar-menu-sub-button\"\n data-sidebar=\"menu-sub-button\"\n data-size={size}\n data-active={isActive}\n className={cn(\n 'text-sidebar-foreground',\n 'ring-sidebar-ring',\n 'flex h-7 min-w-0 -translate-x-px items-center gap-2 overflow-hidden rounded-md px-2 outline-hidden',\n 'focus-visible:ring-2',\n 'hover:bg-sidebar-accent',\n 'hover:text-sidebar-accent-foreground',\n 'active:bg-sidebar-accent',\n 'active:text-sidebar-accent-foreground',\n 'disabled:pointer-events-none',\n 'disabled:opacity-50',\n 'aria-disabled:pointer-events-none',\n 'aria-disabled:opacity-50',\n '[&>svg]:size-4',\n '[&>svg]:shrink-0',\n '[&>svg]:text-sidebar-accent-foreground',\n '[&>span:last-child]:truncate',\n 'data-[active=true]:bg-sidebar-accent',\n 'data-[active=true]:text-sidebar-accent-foreground',\n size === 'sm' && 'text-xs',\n size === 'md' && 'text-sm',\n 'group-data-[collapsible=icon]:hidden',\n className\n )}\n {...props}\n />\n );\n}\n\nexport {\n ServiceLayoutSidebar,\n ServiceLayoutSidebarContent,\n ServiceLayoutSidebarFooter,\n ServiceLayoutSidebarGroup,\n ServiceLayoutSidebarGroupAction,\n ServiceLayoutSidebarGroupContent,\n ServiceLayoutSidebarGroupLabel,\n ServiceLayoutSidebarHeader,\n ServiceLayoutSidebarInset,\n ServiceLayoutSidebarMenu,\n ServiceLayoutSidebarMenuAction,\n ServiceLayoutSidebarMenuBadge,\n ServiceLayoutSidebarMenuButton,\n ServiceLayoutSidebarMenuItem,\n ServiceLayoutSidebarMenuSkeleton,\n ServiceLayoutSidebarMenuSub,\n ServiceLayoutSidebarMenuSubButton,\n ServiceLayoutSidebarMenuSubItem,\n ServiceLayoutSidebarProvider,\n ServiceLayoutSidebarRail,\n ServiceLayoutSidebarSeparator,\n ServiceLayoutSidebarTrigger,\n // biome-ignore lint/style/useComponentExportOnlyModules: true\n useServiceLayoutSidebar,\n};\n","'use client';\nimport { useState } from 'react';\n\nimport { LockIcon, LogInIcon, LogOutIcon, ShoppingBag, ShoppingCartIcon, UserIcon } from 'lucide-react';\n\nimport { useMediaQuery } from '@customafk/react-toolkit/hooks/useMediaQuery';\nimport { cn } from '@customafk/react-toolkit/utils';\n\nimport { Avatar, AvatarFallback, AvatarImage } from '@/components/ui/avatar';\nimport { Button } from '@/components/ui/button';\nimport { Dialog, DialogContent, DialogFooter, DialogHeader, DialogTitle } from '@/components/ui/dialog';\nimport { Drawer, DrawerContent, DrawerFooter, DrawerHeader, DrawerTitle } from '@/components/ui/drawer';\nimport {\n DropdownMenu,\n DropdownMenuContent,\n DropdownMenuItem,\n DropdownMenuLabel,\n DropdownMenuSeparator,\n DropdownMenuTrigger,\n} from '@/components/ui/dropdown-menu';\nimport { Separator } from '@/components/ui/separator';\nimport { Sheet, SheetContent, SheetHeader, SheetTitle, SheetTrigger } from '@/components/ui/sheet';\nimport { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs';\n\nimport { GoogleLogin } from '@react-oauth/google';\nimport { CartList } from './components/cart';\nimport { ServiceLayoutContext, type ServiceLayoutContextProps, useServiceLayout } from './hooks/use-service-layout';\nimport {\n ServiceLayoutSidebar as LayoutSidebar,\n ServiceLayoutSidebarInset,\n ServiceLayoutSidebarProvider,\n ServiceLayoutSidebarTrigger,\n} from './service-layout-sidebar';\n\nexport const ServiceLayoutProvider: React.FC<React.PropsWithChildren<ServiceLayoutContextProps>> = ({\n isLoggedIn = false,\n username,\n email,\n inStockCarts,\n orderedCarts,\n onGoogleLoginSuccess,\n onUpdatingCart,\n onDeletingCart,\n onLogout,\n children,\n}) => {\n return (\n <ServiceLayoutContext.Provider\n value={{\n isLoggedIn,\n username,\n email,\n inStockCarts,\n orderedCarts,\n onGoogleLoginSuccess,\n onUpdatingCart,\n onDeletingCart,\n onLogout,\n }}\n >\n {children}\n </ServiceLayoutContext.Provider>\n );\n};\n\nexport const ServiceLayoutWrapper: React.FC<React.PropsWithChildren> = ({ children }) => {\n return <ServiceLayoutSidebarProvider>{children}</ServiceLayoutSidebarProvider>;\n};\n\ntype ServiceLayoutUserInfoProps = {\n userName?: string;\n userEmail?: string;\n onLogout?: () => void | Promise<void>;\n};\nexport const ServiceLayoutUserInfo: React.FC<ServiceLayoutUserInfoProps> = ({ userName = 'Keith Kennedy', userEmail = 'k.kennedy@originui.com', onLogout }) => {\n return (\n <DropdownMenu>\n <DropdownMenuTrigger asChild>\n <Button size=\"icon\" variant=\"ghost\" color=\"secondary\" className=\"size-10 rounded-full\">\n <Avatar className=\"size-10\">\n <AvatarImage />\n <AvatarFallback className=\"size-full bg-muted-muted\">\n <UserIcon />\n </AvatarFallback>\n </Avatar>\n </Button>\n </DropdownMenuTrigger>\n <DropdownMenuContent align=\"end\" className=\"max-w-64\">\n <DropdownMenuLabel className=\"flex min-w-0 flex-col\">\n <span className=\"truncate font-medium text-sm text-text-positive\">{userName}</span>\n <span className=\"truncate font-normal text-text-positive-weak text-xs\">{userEmail}</span>\n </DropdownMenuLabel>\n <DropdownMenuSeparator />\n <DropdownMenuItem onClick={onLogout}>\n <LogOutIcon size={16} aria-hidden=\"true\" />\n <span className=\"text-text-positive\">Logout</span>\n </DropdownMenuItem>\n </DropdownMenuContent>\n </DropdownMenu>\n );\n};\n\nexport const ServiceLayoutCartInfo: React.FC = () => {\n const { inStockCarts = [], orderedCarts = [] } = useServiceLayout();\n const inStockCount = inStockCarts.length;\n const preOrderCount = orderedCarts.length;\n const totalItems = inStockCount + preOrderCount;\n\n return (\n <Sheet>\n <SheetTrigger asChild>\n <Button size=\"icon\" variant=\"ghost\" color=\"secondary\" className=\"relative size-10 rounded-full\">\n <ShoppingCartIcon />\n </Button>\n </SheetTrigger>\n <SheetContent className=\"w-[95vw] sm:max-w-md\">\n <SheetHeader className=\"flex-0 border-border-weak border-b pb-3\">\n <SheetTitle className=\"flex items-center gap-2\">\n <ShoppingBag size={20} />\n <span>Giỏ hàng của bạn</span>\n {totalItems > 0 && <span className=\"font-normal text-sm text-text-positive-weak\">({totalItems} sản phẩm)</span>}\n </SheetTitle>\n </SheetHeader>\n <Tabs defaultValue={preOrderCount > 0 ? 'pre_order' : 'in_stock'} className=\"h-full flex-1 overflow-y-auto p-4 pt-0\">\n <TabsList className=\"w-full flex-0\">\n <TabsTrigger value=\"in_stock\" className=\"relative\">\n Có sẵn\n {inStockCount > 0 && <span>({inStockCount})</span>}\n </TabsTrigger>\n <TabsTrigger value=\"pre_order\" className=\"relative\">\n Đặt trước\n {preOrderCount > 0 && <span>({preOrderCount})</span>}\n </TabsTrigger>\n </TabsList>\n <TabsContent value=\"pre_order\" className=\"flex-1 overflow-y-auto\">\n <CartList items={orderedCarts} cartType=\"pre_order\" />\n </TabsContent>\n <TabsContent value=\"in_stock\" className=\"flex-1 overflow-y-auto\">\n <CartList items={inStockCarts} cartType=\"in_stock\" />\n </TabsContent>\n </Tabs>\n </SheetContent>\n </Sheet>\n );\n};\n\ntype ServiceLayoutHeaderProps = {\n isLoggedIn?: boolean;\n};\nexport const ServiceLayoutHeader: React.FC<ServiceLayoutHeaderProps> = () => {\n const { isLoggedIn, username, email, onGoogleLoginSuccess, onLogout } = useServiceLayout();\n\n const isDesktop = useMediaQuery('(min-width: 640px)');\n\n const [loginOpen, setLoginOpen] = useState<boolean>(false);\n\n return (\n <header\n className={cn(\n 'bg-card',\n 'h-(--header-height)',\n 'sm:h-[calc(var(--header-height)+0.5rem)] sm:px-4 sm:pr-6',\n 'absolute inset-x-0 top-0 z-20 gap-2 px-2 pr-4.5',\n 'flex items-center shadow-nav',\n 'transition-[width,height] ease-linear'\n )}\n >\n {isLoggedIn && <ServiceLayoutSidebarTrigger />}\n\n <div className=\"flex gap-x-2 sm:ml-2.5\">\n <div className=\"flex aspect-square size-8 items-center justify-center rounded-lg bg-sidebar-primary text-sidebar-primary-foreground\">\n <ShoppingCartIcon size={20} />\n </div>\n <div className=\"grid flex-1 text-left text-sm leading-tight\">\n <span className=\"truncate font-medium\">Lunas Store</span>\n <span className=\"truncate text-xs\">Established 2023</span>\n </div>\n </div>\n\n <div className=\"flex flex-1 items-center justify-end gap-x-2\">\n {!isLoggedIn && (\n <Button className=\"w-8 sm:w-fit\" onClick={() => setLoginOpen(true)}>\n <LogInIcon />\n <span className=\"sr-only sm:not-sr-only\">Đăng nhập</span>\n </Button>\n )}\n {isLoggedIn && (\n <>\n <ServiceLayoutUserInfo userName={username} userEmail={email} onLogout={onLogout} />\n <Separator orientation=\"vertical\" className=\"min-h-6 w-px\" />\n <ServiceLayoutCartInfo />\n </>\n )}\n </div>\n\n {isDesktop && (\n <Dialog open={loginOpen} onOpenChange={setLoginOpen}>\n <DialogContent showCloseButton={false} className=\"flex flex-col gap-0 border-none p-0 sm:max-w-sm\">\n <DialogHeader className=\"flex-0 gap-2 p-6\">\n <DialogTitle className=\"text-center\">Chào mừng bạn đến với Lunas Store!</DialogTitle>\n </DialogHeader>\n <div className=\"flex flex-1 flex-col\">\n <main className=\"size-full flex-1 bg-card p-4 pt-0\">\n <div className=\"flex flex-col items-center gap-y-1\">\n <p className=\"text-sm text-text-positive-weak\">Đăng nhập với Google</p>\n <GoogleLogin\n size=\"large\"\n theme=\"outline\"\n width={240}\n onSuccess={async response => {\n if (!response.clientId || !response.credential || !response.select_by) return;\n await onGoogleLoginSuccess?.(response);\n }}\n />\n </div>\n </main>\n </div>\n <DialogFooter className=\"p-2\" />\n </DialogContent>\n </Dialog>\n )}\n\n {!isDesktop && (\n <Drawer open={loginOpen} onOpenChange={setLoginOpen}>\n <DrawerContent>\n <DrawerHeader className=\"text-left\">\n <DrawerTitle>Chào mừng bạn đến với Lunas Store!</DrawerTitle>\n </DrawerHeader>\n <div className=\"flex flex-1 flex-col\">\n <main className=\"flex size-full flex-1 flex-col p-4 pt-0\">\n <div className=\"flex flex-col items-center gap-y-1\">\n <p className=\"text-sm text-text-positive-weak\">Đăng nhập với Google</p>\n <GoogleLogin\n size=\"large\"\n theme=\"outline\"\n onSuccess={async response => {\n if (!response.clientId || !response.credential || !response.select_by) return;\n await onGoogleLoginSuccess?.(response);\n }}\n />\n </div>\n </main>\n </div>\n <DrawerFooter />\n </DrawerContent>\n </Drawer>\n )}\n </header>\n );\n};\n\nexport const ServiceLayoutSidebar: React.FC<React.PropsWithChildren & React.ComponentProps<typeof LayoutSidebar>> = ({ children, ...props }) => {\n const { isLoggedIn } = useServiceLayout();\n if (!isLoggedIn) return null;\n return (\n <LayoutSidebar variant=\"inset\" collapsible=\"icon\" {...props}>\n {children}\n </LayoutSidebar>\n );\n};\n\nexport const ServiceLayoutMain: React.FC<React.PropsWithChildren> = ({ children }) => {\n const { isLoggedIn, onGoogleLoginSuccess } = useServiceLayout();\n const isDesktop = useMediaQuery('(min-width: 640px)');\n const [loginOpen, setLoginOpen] = useState<boolean>(false);\n\n if (!isLoggedIn) {\n return (\n <div className=\"size-full p-4 pt-[calc(var(--header-height)+1.5rem)]\">\n <div className=\"flex size-full flex-col items-center justify-center gap-6 rounded-lg bg-card p-8 text-center shadow-card\">\n <div className=\"flex size-20 items-center justify-center rounded-full bg-muted-foreground/10\">\n <LockIcon className=\"size-10 text-primary\" />\n </div>\n <div className=\"flex max-w-md flex-col gap-2\">\n <h2 className=\"font-semibold text-2xl\">Bạn chưa đăng nhập</h2>\n <p className=\"text-text-positive-weak\">Đăng nhập để khám phá đầy đủ các tính năng của Lunas Store và truy cập vào tài khoản của bạn.</p>\n </div>\n <Button size=\"lg\" className=\"gap-2\" onClick={() => setLoginOpen(true)}>\n <LogInIcon size={18} />\n <span>Đăng nhập ngay</span>\n </Button>\n\n {isDesktop && (\n <Dialog open={loginOpen} onOpenChange={setLoginOpen}>\n <DialogContent showCloseButton={false} className=\"flex flex-col gap-0 border-none p-0 sm:max-w-sm\">\n <DialogHeader className=\"flex-0 gap-2 p-6\">\n <DialogTitle className=\"text-center\">Chào mừng bạn đến với Lunas Store!</DialogTitle>\n </DialogHeader>\n <div className=\"flex flex-1 flex-col\">\n <main className=\"size-full flex-1 bg-card p-4 pt-0\">\n <div className=\"flex flex-col items-center gap-y-1\">\n <p className=\"text-sm text-text-positive-weak\">Đăng nhập với Google</p>\n <GoogleLogin\n size=\"large\"\n theme=\"outline\"\n width={240}\n onSuccess={async response => {\n if (!response.clientId || !response.credential || !response.select_by) return;\n await onGoogleLoginSuccess?.(response);\n setLoginOpen(false);\n }}\n />\n </div>\n </main>\n </div>\n <DialogFooter className=\"p-2\" />\n </DialogContent>\n </Dialog>\n )}\n\n {!isDesktop && (\n <Drawer open={loginOpen} onOpenChange={setLoginOpen}>\n <DrawerContent>\n <DrawerHeader className=\"text-left\">\n <DrawerTitle>Chào mừng bạn đến với Lunas Store!</DrawerTitle>\n </DrawerHeader>\n <div className=\"flex flex-1 flex-col\">\n <main className=\"flex size-full flex-1 flex-col p-4 pt-0\">\n <div className=\"flex flex-col items-center gap-y-1\">\n <p className=\"text-sm text-text-positive-weak\">Đăng nhập với Google</p>\n <GoogleLogin\n size=\"large\"\n theme=\"outline\"\n onSuccess={async response => {\n if (!response.clientId || !response.credential || !response.select_by) return;\n await onGoogleLoginSuccess?.(response);\n setLoginOpen(false);\n }}\n />\n </div>\n </main>\n </div>\n <DrawerFooter />\n </DrawerContent>\n </Drawer>\n )}\n </div>\n </div>\n );\n }\n\n return (\n <ServiceLayoutSidebarInset>\n <section className=\"relative size-full\">\n <div className=\"absolute inset-0 flex flex-col\">{children}</div>\n </section>\n </ServiceLayoutSidebarInset>\n );\n};\nexport const ServiceLayoutMainHeader: React.FC<React.PropsWithChildren & { className?: string }> = ({ className, children }) => {\n return (\n <div data-slot=\"main-header\" className={cn('flex-0 snap-start', className)}>\n {children}\n </div>\n );\n};\n\nexport const ServiceLayoutMainContent: React.FC<React.PropsWithChildren & { className?: string }> = ({ className, children }) => {\n return (\n <div data-slot=\"main-content\" className={cn('flex w-full flex-1 flex-col gap-4 overflow-y-auto px-2 sm:px-4', className)}>\n {children}\n </div>\n );\n};\n\nexport const ServiceLayoutMainFooter: React.FC<React.PropsWithChildren & { className?: string }> = ({ className, children }) => {\n return (\n <div data-slot=\"main-footer\" className={cn('hidden w-full flex-0 border-border-weak border-t pt-2 sm:flex', className)}>\n {children}\n </div>\n );\n};\n\nexport const ServiceLayoutMainGroup: React.FC<React.PropsWithChildren<{ className?: string }>> = ({ children, className }) => {\n return (\n <div data-slot=\"main-group\" className={cn('flex size-full flex-col gap-4', className)}>\n {children}\n </div>\n );\n};\n\nexport const ServiceLayoutMainGroupContent: React.FC<React.PropsWithChildren & { className?: string }> = ({ className, children }) => {\n return (\n <div data-slot=\"main-group-content\" className={cn('size-full max-w-8xl flex-1 rounded-md bg-card p-4 shadow-card', className)}>\n {children}\n </div>\n );\n};\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.mjs","names":["CartItem: React.FC<\n CartItemProps & {\n type: 'in_stock' | 'pre_order';\n }\n>","Image","CartList: React.FC<CartListProps>","ScrollArea","Separator","open","ServiceLayoutSidebar","SlotPrimitive","Tooltip","ServiceLayoutProvider: React.FC<React.PropsWithChildren<ServiceLayoutContextProps>>","ServiceLayoutWrapper: React.FC<React.PropsWithChildren>","ServiceLayoutUserInfo: React.FC<ServiceLayoutUserInfoProps>","DropdownMenu","Avatar","ServiceLayoutCartInfo: React.FC","Tabs","ServiceLayoutHeader: React.FC<ServiceLayoutHeaderProps>","Separator","Dialog","ServiceLayoutSidebar: React.FC<React.PropsWithChildren & React.ComponentProps<typeof LayoutSidebar>>","LayoutSidebar","ServiceLayoutMain: React.FC<React.PropsWithChildren>","ServiceLayoutMainHeader: React.FC<React.PropsWithChildren & { className?: string }>","ServiceLayoutMainContent: React.FC<React.PropsWithChildren & { className?: string }>","ServiceLayoutMainFooter: React.FC<React.PropsWithChildren & { className?: string }>","ServiceLayoutMainGroup: React.FC<React.PropsWithChildren<{ className?: string }>>","ServiceLayoutMainGroupContent: React.FC<React.PropsWithChildren & { className?: string }>"],"sources":["../../../packages/components/layouts/service-layout/hooks/use-service-layout.ts","../../../packages/components/layouts/service-layout/components/cart/index.tsx","../../../packages/components/layouts/service-layout/hooks/use-service-layout-sidebar.ts","../../../packages/components/layouts/service-layout/service-layout-sidebar.tsx","../../../packages/components/layouts/service-layout/service-layout.tsx"],"sourcesContent":["import { createContext, use } from 'react';\n\nimport type { CredentialResponse } from '@react-oauth/google';\n\ntype Cart = {\n id: string;\n productUuid: string;\n productName: string;\n variantUuid: string;\n variantName: string;\n imageUrl: string;\n options: Array<{ label: string; value: string }>;\n quantity: number;\n price: number;\n};\n\nexport type ServiceLayoutContextProps = {\n isLoggedIn?: boolean;\n username?: string;\n email?: string;\n inStockCarts?: Cart[];\n orderedCarts?: Cart[];\n onGoogleLoginSuccess?: (params: CredentialResponse) => void | Promise<void>;\n onUpdatingCart?: (id: string, quantity: number, type: 'in_stock' | 'pre_order') => void | Promise<void>;\n onDeletingCart?: (id: string) => void | Promise<void>;\n onLogout?: () => void | Promise<void>;\n};\n\nexport const ServiceLayoutContext = createContext<ServiceLayoutContextProps | null>(null);\n\nexport const useServiceLayout = () => {\n const context = use(ServiceLayoutContext);\n if (!context) {\n throw new Error('useServiceLayoutContext must be used within a ServiceLayoutProvider');\n }\n return context;\n};\n","import { useCallback, useEffect, useState } from 'react';\n\nimport { Minus, Plus, ShoppingBasketIcon, Trash2 } from 'lucide-react';\n\nimport { useDebounceCallback } from '@customafk/react-toolkit/hooks/useDebounceCallback';\nimport { cn } from '@customafk/react-toolkit/utils';\n\nimport { Button } from '@/components/ui/button';\nimport { Card, CardContent } from '@/components/ui/card';\nimport { Image } from '@/components/ui/image';\nimport { Input } from '@/components/ui/input';\nimport { ScrollArea } from '@/components/ui/scroll-area';\nimport { Separator } from '@/components/ui/separator';\nimport { useServiceLayout } from '../../hooks/use-service-layout';\n\ntype CartItemProps = {\n id: string;\n productUuid: string;\n productName: string;\n variantUuid: string;\n variantName: string;\n imageUrl: string;\n options: Array<{ label: string; value: string }>;\n quantity: number;\n price: number;\n};\n\nexport const CartItem: React.FC<\n CartItemProps & {\n type: 'in_stock' | 'pre_order';\n }\n> = ({ id, type, productName, imageUrl, quantity, price, options }) => {\n const { onDeletingCart, onUpdatingCart } = useServiceLayout();\n const [itemQuantity, setItemQuantity] = useState(quantity);\n const [isDeleting, setIsDeleting] = useState<boolean>(false);\n\n const handleUpdate = useCallback(() => {\n onUpdatingCart?.(id, itemQuantity, type);\n }, [onUpdatingCart, id, itemQuantity, type]);\n\n const handleUpdateDebounced = useDebounceCallback(handleUpdate, 500);\n\n const handleQuantityChange = useCallback((value: number) => {\n if (value < 1) value = 1;\n if (value > 99) value = 99;\n setItemQuantity(value);\n }, []);\n\n const handleRemoveItem = useCallback(async () => {\n setIsDeleting(true);\n await onDeletingCart?.(id);\n setIsDeleting(false);\n }, [id, onDeletingCart]);\n\n useEffect(() => {\n if (itemQuantity !== quantity) {\n handleUpdateDebounced();\n }\n return () => {\n handleUpdateDebounced.cancel();\n };\n }, [itemQuantity, quantity, handleUpdateDebounced]);\n\n return (\n <Card className=\"border-border-weak relative mb-3 overflow-x-auto border p-4 shadow-none\">\n {isDeleting && (\n <div className=\"bg-muted-muted/80 absolute inset-0 z-10 flex items-center justify-center\">\n <div className=\"loader-dots\" />\n </div>\n )}\n <CardContent className=\"p-0\">\n <Button\n variant=\"ghost\"\n size=\"icon\"\n color=\"muted\"\n disabled={isDeleting}\n className=\"text-text-positive-weak absolute top-2 right-2 z-10\"\n onClick={handleRemoveItem}\n >\n <Trash2 className=\"h-4 w-4\" />\n <span className=\"sr-only\">Remove item</span>\n </Button>\n <div className=\"flex gap-3\">\n <div className=\"relative size-20 flex-shrink-0\">\n {imageUrl ? (\n <Image src={imageUrl} alt={productName} className=\"rounded-md border-none shadow-xs\" width={80} height={80} />\n ) : (\n <div className=\"bg-muted text-muted-foreground flex size-full items-center justify-center text-xs\">No image</div>\n )}\n </div>\n\n <div className=\"flex flex-1 flex-col\">\n <div className=\"flex justify-between\">\n <h3 className=\"text-text-positive line-clamp-1 text-sm font-medium\">{productName}</h3>\n </div>\n\n <p className=\"text-text-positive-weak mb-1 text-xs\">\n {options.map((option, index) => (\n <span key={option.label}>\n {option.label}: {option.value}\n {index < options.length - 1 && ', '}\n </span>\n ))}\n </p>\n\n <div className=\"mt-auto flex items-center justify-between\">\n <div className=\"flex items-center space-x-1\">\n <button\n disabled={itemQuantity <= 1}\n className=\"border-border active:bg-muted-muted flex size-6 cursor-pointer items-center justify-center rounded-full border transition-all disabled:opacity-60\"\n onClick={() => handleQuantityChange(itemQuantity - 1)}\n >\n <Minus size={12} />\n </button>\n <Input\n value={itemQuantity}\n onChange={e => handleQuantityChange(parseInt(e.target.value || '1'))}\n className=\"border-border h-6.5 w-14 rounded-md border p-1 text-center text-sm\"\n min={1}\n />\n <button\n disabled={itemQuantity >= 99}\n className=\"border-border active:bg-muted-muted flex size-6 cursor-pointer items-center justify-center rounded-full border transition-all disabled:opacity-60\"\n onClick={() => handleQuantityChange(itemQuantity + 1)}\n >\n <Plus size={12} />\n </button>\n </div>\n\n <div className=\"text-sm font-semibold\">{(price * itemQuantity).toLocaleString()} ₫</div>\n </div>\n </div>\n </div>\n </CardContent>\n </Card>\n );\n};\n\ntype CartListProps = {\n items?: CartItemProps[];\n cartType: 'in_stock' | 'pre_order';\n className?: string;\n};\n\nexport const CartList: React.FC<CartListProps> = ({ items = [], cartType, className }) => {\n const totalAmount = items.reduce((sum, item) => sum + item.price * item.quantity, 0);\n\n if (items.length === 0) {\n return (\n <div className={cn('bg-muted-muted flex size-full max-h-80 flex-col items-center justify-center rounded-lg p-8', className)}>\n <div className=\"text-text-positive-weak bg-card shadow-card mb-4 flex size-20 items-center justify-center rounded-full text-5xl\">\n <ShoppingBasketIcon size={42} strokeWidth={1} />\n </div>\n <div className=\"flex flex-col space-y-1\">\n <p className=\"text-text-positive text-center text-base font-semibold\">Giỏ hàng trống</p>\n <p className=\"text-text-positive-weak mb-4 text-center text-sm\">\n {cartType === 'in_stock' ? 'Thêm sản phẩm có sẵn vào giỏ hàng của bạn!' : 'Thêm sản phẩm đặt trước vào giỏ hàng của bạn!'}\n </p>\n </div>\n </div>\n );\n }\n\n return (\n <div className={cn('flex h-full flex-col space-y-4 overflow-y-auto', className)}>\n <ScrollArea className=\"h-full flex-1\">\n {items.map(item => (\n <CartItem key={item.id} {...item} type={cartType} />\n ))}\n </ScrollArea>\n\n <div className=\"flex-0 space-y-3\">\n <div className=\"flex justify-between\">\n <span className=\"text-text-positive-weak text-sm\">Tạm tính:</span>\n <span className=\"text-sm\">{totalAmount.toLocaleString()} ₫</span>\n </div>\n <Separator />\n <div className=\"flex justify-between\">\n <span className=\"text-text-positive font-medium\">Tổng cộng:</span>\n <span className=\"text-lg font-semibold\">{totalAmount.toLocaleString()} ₫</span>\n </div>\n <Button className=\"mt-2 w-full\">Thanh toán ngay</Button>\n {cartType === 'pre_order' && <p className=\"text-text-positive-weak text-center text-xs italic\">* Sản phẩm đặt trước sẽ được giao sau khi có hàng</p>}\n </div>\n </div>\n );\n};\n","import { createContext, use } from 'react';\n\nexport type SidebarContextProps = {\n state: 'expanded' | 'collapsed';\n open: boolean;\n setOpen: (open: boolean) => void;\n openMobile: boolean;\n setOpenMobile: (open: boolean) => void;\n isMobile: boolean;\n toggleSidebar: () => void;\n};\n\nexport const SidebarContext = createContext<SidebarContextProps | null>(null);\n\nexport const useServiceLayoutSidebar = () => {\n const context = use(SidebarContext);\n if (!context) {\n throw new Error('useSidebar must be used within a SidebarProvider.');\n }\n\n return context;\n};\n","import { useCallback, useEffect, useMemo, useState } from 'react';\n\nimport { LogOutIcon, MenuIcon, ShoppingCartIcon } from 'lucide-react';\n\nimport { useIsMobile } from '@customafk/react-toolkit/hooks/useMobile';\nimport { cn } from '@customafk/react-toolkit/utils';\n\nimport { Button } from '@/components/ui/button';\nimport { Separator } from '@/components/ui/separator';\nimport { Sheet, SheetContent, SheetDescription, SheetHeader, SheetTitle } from '@/components/ui/sheet';\nimport { Skeleton } from '@/components/ui/skeleton';\nimport { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';\n\nimport { cva, type VariantProps } from 'class-variance-authority';\nimport { Slot as SlotPrimitive } from 'radix-ui';\nimport { useServiceLayout } from './hooks/use-service-layout';\nimport { SidebarContext, type SidebarContextProps, useServiceLayoutSidebar } from './hooks/use-service-layout-sidebar';\n\nconst SIDEBAR_COOKIE_NAME = 'sidebar_state';\nconst SIDEBAR_COOKIE_MAX_AGE = 60 * 60 * 24 * 7;\nconst SIDEBAR_WIDTH = '16rem';\nconst SIDEBAR_WIDTH_MOBILE = '16rem';\nconst SIDEBAR_WIDTH_ICON = '3rem';\nconst SIDEBAR_KEYBOARD_SHORTCUT = 'b';\n\nfunction ServiceLayoutSidebarProvider({\n defaultOpen = true,\n open: openProp,\n onOpenChange: setOpenProp,\n className,\n style,\n children,\n ...props\n}: React.ComponentProps<'div'> & {\n defaultOpen?: boolean;\n open?: boolean;\n onOpenChange?: (open: boolean) => void;\n}) {\n const isMobile = useIsMobile();\n const [openMobile, setOpenMobile] = useState(false);\n\n // This is the internal state of the sidebar.\n // We use openProp and setOpenProp for control from outside the component.\n const [_open, _setOpen] = useState(defaultOpen);\n const open = openProp ?? _open;\n const setOpen = useCallback(\n (value: boolean | ((value: boolean) => boolean)) => {\n const openState = typeof value === 'function' ? value(open) : value;\n if (setOpenProp) {\n setOpenProp(openState);\n } else {\n _setOpen(openState);\n }\n\n // This sets the cookie to keep the sidebar state.\n // biome-ignore lint/suspicious/noDocumentCookie: all\n document.cookie = `${SIDEBAR_COOKIE_NAME}=${openState}; path=/; max-age=${SIDEBAR_COOKIE_MAX_AGE}`;\n },\n [setOpenProp, open]\n );\n\n // Helper to toggle the sidebar.\n const toggleSidebar = useCallback(() => {\n return isMobile ? setOpenMobile(open => !open) : setOpen(open => !open);\n }, [isMobile, setOpen]);\n\n // Adds a keyboard shortcut to toggle the sidebar.\n useEffect(() => {\n const handleKeyDown = (event: KeyboardEvent) => {\n if (event.key === SIDEBAR_KEYBOARD_SHORTCUT && (event.metaKey || event.ctrlKey)) {\n event.preventDefault();\n toggleSidebar();\n }\n };\n\n window.addEventListener('keydown', handleKeyDown);\n return () => window.removeEventListener('keydown', handleKeyDown);\n }, [toggleSidebar]);\n\n // We add a state so that we can do data-state=\"expanded\" or \"collapsed\".\n // This makes it easier to style the sidebar with Tailwind classes.\n const state = open ? 'expanded' : 'collapsed';\n\n const contextValue = useMemo<SidebarContextProps>(\n () => ({\n state,\n isMobile,\n\n toggleSidebar,\n\n open,\n setOpen,\n\n openMobile,\n setOpenMobile,\n }),\n [state, open, setOpen, isMobile, openMobile, toggleSidebar]\n );\n\n return (\n <SidebarContext.Provider value={contextValue}>\n <TooltipProvider delayDuration={0}>\n <div\n data-slot=\"sidebar-wrapper\"\n style={\n {\n '--sidebar-width': SIDEBAR_WIDTH,\n '--sidebar-width-icon': SIDEBAR_WIDTH_ICON,\n ...style,\n } as React.CSSProperties\n }\n className={cn('group/sidebar-wrapper', 'has-data-[variant=inset]:bg-sidebar', 'flex h-dvh w-full', className)}\n {...props}\n >\n {children}\n </div>\n </TooltipProvider>\n </SidebarContext.Provider>\n );\n}\n\nfunction ServiceLayoutSidebar({\n side = 'left',\n variant = 'sidebar',\n collapsible = 'offcanvas',\n className,\n children,\n ...props\n}: React.ComponentProps<'div'> & {\n side?: 'left' | 'right';\n variant?: 'sidebar' | 'floating' | 'inset';\n collapsible?: 'offcanvas' | 'icon' | 'none';\n}) {\n const { isMobile, state, openMobile, setOpenMobile } = useServiceLayoutSidebar();\n\n if (collapsible === 'none') {\n return (\n <aside data-slot=\"sidebar\" className={cn('flex h-full w-(--sidebar-width) flex-col bg-sidebar text-sidebar-foreground', className)} {...props}>\n {children}\n </aside>\n );\n }\n\n if (isMobile) {\n return (\n <Sheet open={openMobile} onOpenChange={setOpenMobile}>\n <SheetContent\n side=\"left\"\n data-sidebar=\"sidebar\"\n data-slot=\"sidebar\"\n data-mobile=\"true\"\n className=\"w-(--sidebar-width) bg-sidebar p-0 text-sidebar-foreground sm:max-w-3xs [&>button]:hidden\"\n style={\n {\n '--sidebar-width': SIDEBAR_WIDTH_MOBILE,\n } as React.CSSProperties\n }\n >\n <SheetHeader className=\"sr-only\">\n <SheetTitle>Sidebar</SheetTitle>\n <SheetDescription>Displays the mobile sidebar.</SheetDescription>\n </SheetHeader>\n <div className=\"flex size-full flex-col\">\n <div className=\"flex flex-0 items-center gap-x-2 border-border-weak border-b p-2 pr-4\">\n <ServiceLayoutSidebarTrigger />\n <div className=\"ml-2 flex aspect-square size-8 items-center justify-center rounded-lg bg-sidebar-primary text-sidebar-primary-foreground\">\n <ShoppingCartIcon size={20} />\n </div>\n <div className=\"grid flex-1 text-left text-sm leading-tight\">\n <span className=\"truncate font-medium\">Lunas Store</span>\n <span className=\"truncate text-xs\">Established 2023</span>\n </div>\n </div>\n <div className=\"flex flex-1 flex-col p-2\">{children}</div>\n </div>\n </SheetContent>\n </Sheet>\n );\n }\n\n return (\n <aside\n className=\"group peer hidden bg-card text-sidebar-foreground md:block\"\n data-state={state}\n data-collapsible={state === 'collapsed' ? collapsible : ''}\n data-variant={variant}\n data-side={side}\n data-slot=\"sidebar\"\n >\n {/* This is what handles the sidebar gap on desktop */}\n <div\n data-slot=\"sidebar-gap\"\n className={cn(\n 'relative',\n 'bg-transparent',\n 'transition-[width] duration-200 ease-linear',\n 'h-(--header-height) w-(--sidebar-width)',\n 'sm:h-[calc(var(--header-height)+0.5rem)]',\n 'group-data-[collapsible=offcanvas]:w-0',\n 'group-data-[side=right]:rotate-180',\n variant === 'floating' || variant === 'inset'\n ? 'group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4)))]'\n : 'group-data-[collapsible=icon]:w-(--sidebar-width-icon)'\n )}\n />\n <div\n data-slot=\"sidebar-container\"\n className={cn(\n 'hidden md:flex',\n 'fixed inset-y-0 top-14 z-10 shadow-nav',\n 'h-[calc(100dvh-3.5rem)] w-(--sidebar-width)',\n 'transition-[left,right,width] duration-200 ease-linear',\n side === 'left' && 'left-0 group-data-[collapsible=offcanvas]:left-[calc(var(--sidebar-width)*-1)]',\n side === 'right' && 'right-0 group-data-[collapsible=offcanvas]:right-[calc(var(--sidebar-width)*-1)]',\n // Adjust the padding for floating and inset variants.\n variant === 'floating' || variant === 'inset'\n ? 'p-2 group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4)))]'\n : 'group-data-[collapsible=icon]:w-(--sidebar-width-icon) group-data-[side=left]:border-r group-data-[side=right]:border-l',\n className\n )}\n {...props}\n >\n <div\n data-sidebar=\"sidebar\"\n data-slot=\"sidebar-inner\"\n className={cn(\n 'flex size-full flex-col',\n 'group-data-[variant=floating]:rounded-lg',\n 'group-data-[variant=floating]:border',\n 'group-data-[variant=floating]:border-sidebar-border',\n 'group-data-[variant=floating]:shadow-sm'\n )}\n >\n {children}\n </div>\n </div>\n </aside>\n );\n}\n\nfunction ServiceLayoutSidebarTrigger({ className, onClick, ...props }: React.ComponentProps<typeof Button>) {\n const { toggleSidebar } = useServiceLayoutSidebar();\n\n return (\n <Button\n data-sidebar=\"trigger\"\n data-slot=\"sidebar-trigger\"\n variant=\"ghost\"\n color=\"muted\"\n size=\"icon\"\n className={cn('size-10 rounded-full', className)}\n onClick={event => {\n onClick?.(event);\n toggleSidebar();\n }}\n {...props}\n >\n <MenuIcon className=\"size-6!\" />\n <span className=\"sr-only\">Toggle Sidebar</span>\n </Button>\n );\n}\n\nfunction ServiceLayoutSidebarRail({ className, ...props }: React.ComponentProps<'button'>) {\n const { toggleSidebar } = useServiceLayoutSidebar();\n\n return (\n <button\n data-sidebar=\"rail\"\n data-slot=\"sidebar-rail\"\n aria-label=\"Toggle Sidebar\"\n tabIndex={-1}\n onClick={toggleSidebar}\n title=\"Toggle Sidebar\"\n className={cn(\n 'absolute inset-y-0 z-20 hidden w-4 -translate-x-1/2 transition-all ease-linear sm:flex',\n 'after:absolute after:inset-y-0 after:left-1/2 after:w-0.5',\n 'group-data-[side=left]:-right-4',\n 'group-data-[side=right]:left-0',\n 'in-data-[side=left]:cursor-w-resize',\n 'in-data-[side=right]:cursor-e-resize',\n 'hover:after:bg-sidebar-border',\n 'hover:group-data-[collapsible=offcanvas]:bg-sidebar',\n 'group-data-[collapsible=offcanvas]:translate-x-0',\n 'group-data-[collapsible=offcanvas]:after:left-full',\n '[[data-side=left][data-state=collapsed]_&]:cursor-e-resize',\n '[[data-side=right][data-state=collapsed]_&]:cursor-w-resize',\n '[[data-side=left][data-collapsible=offcanvas]_&]:-right-2',\n '[[data-side=right][data-collapsible=offcanvas]_&]:-left-2',\n className\n )}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarInset({ className, children, ...props }: React.ComponentProps<'main'>) {\n return (\n <main data-slot=\"sidebar-inset\" className={cn('relative flex w-full flex-1 flex-col', className)} {...props}>\n <div className=\"h-(--header-height) w-full sm:h-[calc(var(--header-height)+0.5rem)]\" />\n <div className={cn('inset-shadow-sm flex-1')}>{children}</div>\n </main>\n );\n}\n\nfunction ServiceLayoutSidebarHeader({ className, ...props }: React.ComponentProps<'div'>) {\n return <div data-slot=\"sidebar-header\" data-sidebar=\"header\" className={cn('flex flex-col gap-2 p-2', className)} {...props} />;\n}\n\nfunction ServiceLayoutSidebarFooter({ className, children, ...props }: React.ComponentProps<'div'>) {\n const { open } = useServiceLayoutSidebar();\n const { onLogout } = useServiceLayout();\n return (\n <div data-slot=\"sidebar-footer\" data-sidebar=\"footer\" className={cn('flex flex-col gap-2', className)} {...props}>\n {children}\n <ServiceLayoutSidebarMenu>\n <ServiceLayoutSidebarMenuItem>\n <ServiceLayoutSidebarMenuButton className=\"border border-border\" onClick={onLogout}>\n <LogOutIcon className=\"text-text-positive-weak\" />\n Đăng xuất\n </ServiceLayoutSidebarMenuButton>\n </ServiceLayoutSidebarMenuItem>\n {open && (\n <ServiceLayoutSidebarMenuItem className=\"mt-2 border-t border-t-border\">\n <p className=\"pt-2 text-center text-muted-foreground text-xs\">Copyright © 2025, Lunas.</p>\n </ServiceLayoutSidebarMenuItem>\n )}\n </ServiceLayoutSidebarMenu>\n </div>\n );\n}\n\nfunction ServiceLayoutSidebarSeparator({ className, ...props }: React.ComponentProps<typeof Separator>) {\n return <Separator data-slot=\"sidebar-separator\" data-sidebar=\"separator\" className={cn('mx-2 w-auto bg-sidebar-border', className)} {...props} />;\n}\n\nfunction ServiceLayoutSidebarContent({ className, ...props }: React.ComponentProps<'div'>) {\n return (\n <div\n data-slot=\"sidebar-content\"\n data-sidebar=\"content\"\n className={cn('flex min-h-0 flex-1 flex-col gap-2 overflow-auto group-data-[collapsible=icon]:overflow-hidden', className)}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarGroup({ className, ...props }: React.ComponentProps<'div'>) {\n return <div data-slot=\"sidebar-group\" data-sidebar=\"group\" className={cn('relative flex w-full min-w-0 flex-col', className)} {...props} />;\n}\n\nfunction ServiceLayoutSidebarGroupLabel({ className, asChild = false, ...props }: React.ComponentProps<'div'> & { asChild?: boolean }) {\n const Comp = asChild ? SlotPrimitive.Slot : 'div';\n\n return (\n <Comp\n data-slot=\"sidebar-group-label\"\n data-sidebar=\"group-label\"\n className={cn(\n 'flex h-8 shrink-0 items-center rounded-md px-2 font-medium text-xs outline-hidden transition-[margin,opacity] duration-200 ease-linear focus-visible:ring-2',\n 'text-sidebar-foreground/70',\n 'ring-sidebar-ring',\n '[&>svg]:size-4',\n '[&>svg]:shrink-0',\n 'group-data-[collapsible=icon]:-mt-8',\n 'group-data-[collapsible=icon]:opacity-0',\n className\n )}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarGroupAction({ className, asChild = false, ...props }: React.ComponentProps<'button'> & { asChild?: boolean }) {\n const Comp = asChild ? SlotPrimitive.Slot : 'button';\n\n return (\n <Comp\n data-slot=\"sidebar-group-action\"\n data-sidebar=\"group-action\"\n className={cn(\n 'absolute top-3.5 right-3 flex aspect-square w-5 items-center justify-center rounded-md p-0 text-sidebar-foreground outline-hidden ring-sidebar-ring transition-transform hover:bg-sidebar-accent hover:text-sidebar-accent-foreground',\n 'focus-visible:ring-2',\n '[&>svg]:size-4',\n '[&>svg]:shrink-0',\n // Increases the hit area of the button on mobile.\n 'after:absolute after:-inset-2 md:after:hidden',\n 'group-data-[collapsible=icon]:hidden',\n className\n )}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarGroupContent({ className, ...props }: React.ComponentProps<'div'>) {\n return <div data-slot=\"sidebar-group-content\" data-sidebar=\"group-content\" className={cn('w-full text-sm', className)} {...props} />;\n}\n\nfunction ServiceLayoutSidebarMenu({ className, ...props }: React.ComponentProps<'ul'>) {\n return <ul data-slot=\"sidebar-menu\" data-sidebar=\"menu\" className={cn('flex w-full min-w-0 flex-col gap-1', className)} {...props} />;\n}\n\nfunction ServiceLayoutSidebarMenuItem({ className, ...props }: React.ComponentProps<'li'>) {\n return <li data-slot=\"sidebar-menu-item\" data-sidebar=\"menu-item\" className={cn('group/menu-item relative', className)} {...props} />;\n}\n\nconst sidebarMenuButtonVariants = cva(\n [\n 'peer/menu-button',\n 'cursor-pointer',\n 'flex w-full items-center gap-2',\n 'overflow-hidden rounded-md p-2 outline-hidden',\n 'truncate text-left',\n 'transition-[color,width,height,padding]',\n 'hover:bg-sidebar-accent',\n 'hover:text-sidebar-accent-foreground',\n 'active:bg-sidebar-accent',\n 'active:text-sidebar-accent-foreground',\n 'disabled:pointer-events-none',\n 'disabled:opacity-50',\n 'group-has-data-[sidebar=menu-action]/menu-item:pr-8',\n 'aria-disabled:pointer-events-none',\n 'aria-disabled:opacity-50',\n 'data-[active=true]:bg-sidebar-primary-muted',\n 'data-[active=true]:font-medium',\n 'data-[active=true]:text-sidebar-primary',\n 'data-[state=open]:hover:bg-sidebar-accent',\n 'data-[state=open]:hover:text-sidebar-accent-foreground',\n 'group-data-[collapsible=icon]:size-12!',\n 'group-data-[collapsible=icon]:p-3!',\n 'group-data-[collapsible=icon]:gap-3!',\n '[&>svg]:size-6',\n '[&>svg]:shrink-0',\n '[&>span:last-child]:truncate',\n ],\n {\n variants: {\n variant: {\n default: 'hover:bg-sidebar-accent/60 hover:text-sidebar-accent-foreground/80',\n outline:\n 'bg-background shadow-[0_0_0_1px_hsl(var(--sidebar-border))] hover:bg-sidebar-accent hover:text-sidebar-accent-foreground hover:shadow-[0_0_0_1px_hsl(var(--sidebar-accent))]',\n },\n size: {\n default: 'h-10 text-sm',\n sm: 'h-7 text-xs',\n lg: 'h-12 text-sm group-data-[collapsible=icon]:p-0!',\n },\n },\n defaultVariants: {\n variant: 'default',\n size: 'default',\n },\n }\n);\n\nfunction ServiceLayoutSidebarMenuButton({\n asChild = false,\n isActive = false,\n variant = 'default',\n size = 'default',\n tooltip,\n className,\n ...props\n}: React.ComponentProps<'button'> & {\n asChild?: boolean;\n isActive?: boolean;\n tooltip?: string | React.ComponentProps<typeof TooltipContent>;\n} & VariantProps<typeof sidebarMenuButtonVariants>) {\n const Comp = asChild ? SlotPrimitive.Slot : 'button';\n const { isMobile, state } = useServiceLayoutSidebar();\n\n const button = (\n <Comp\n data-slot=\"sidebar-menu-button\"\n data-sidebar=\"menu-button\"\n data-size={size}\n data-active={isActive}\n className={cn(sidebarMenuButtonVariants({ variant, size }), className)}\n {...props}\n />\n );\n\n if (!tooltip) {\n return button;\n }\n\n if (typeof tooltip === 'string') {\n tooltip = {\n children: tooltip,\n };\n }\n\n return (\n <Tooltip>\n <TooltipTrigger asChild>{button}</TooltipTrigger>\n <TooltipContent side=\"right\" align=\"center\" hidden={state !== 'collapsed' || isMobile} {...tooltip} />\n </Tooltip>\n );\n}\n\nfunction ServiceLayoutSidebarMenuAction({\n className,\n asChild = false,\n showOnHover = false,\n ...props\n}: React.ComponentProps<'button'> & {\n asChild?: boolean;\n showOnHover?: boolean;\n}) {\n const Comp = asChild ? SlotPrimitive.Slot : 'button';\n\n return (\n <Comp\n data-slot=\"sidebar-menu-action\"\n data-sidebar=\"menu-action\"\n className={cn(\n 'absolute top-1.5 right-1 flex aspect-square w-5 items-center justify-center rounded-md p-0 text-sidebar-foreground outline-hidden ring-sidebar-ring transition-transform hover:bg-sidebar-accent hover:text-sidebar-accent-foreground peer-hover/menu-button:text-sidebar-accent-foreground',\n 'focus-visible:ring-2',\n '[&>svg]:size-4',\n '[&>svg]:shrink-0',\n // Increases the hit area of the button on mobile.\n 'after:absolute after:-inset-2 md:after:hidden',\n 'peer-data-[size=sm]/menu-button:top-1',\n 'peer-data-[size=default]/menu-button:top-1.5',\n 'peer-data-[size=lg]/menu-button:top-2.5',\n 'group-data-[collapsible=icon]:hidden',\n showOnHover && 'peer-data-[active=true]/menu-button:text-sidebar-accent-foreground',\n showOnHover && 'group-focus-within/menu-item:opacity-100',\n showOnHover && 'group-hover/menu-item:opacity-100',\n showOnHover && 'data-[state=open]:opacity-100 md:opacity-0',\n className\n )}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarMenuBadge({ className, ...props }: React.ComponentProps<'div'>) {\n return (\n <div\n data-slot=\"sidebar-menu-badge\"\n data-sidebar=\"menu-badge\"\n className={cn(\n 'pointer-events-none absolute right-1 flex h-5 min-w-5 select-none items-center justify-center rounded-md px-1 font-medium text-sidebar-foreground text-xs tabular-nums',\n 'peer-hover/menu-button:text-sidebar-accent-foreground',\n 'peer-data-[active=true]/menu-button:text-sidebar-accent-foreground',\n 'peer-data-[size=sm]/menu-button:top-1',\n 'peer-data-[size=default]/menu-button:top-1.5',\n 'peer-data-[size=lg]/menu-button:top-2.5',\n 'group-data-[collapsible=icon]:hidden',\n className\n )}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarMenuSkeleton({\n className,\n showIcon = false,\n ...props\n}: React.ComponentProps<'div'> & {\n showIcon?: boolean;\n}) {\n // Random width between 50 to 90%.\n const width = useMemo(() => {\n return `${Math.floor(Math.random() * 40) + 50}%`;\n }, []);\n\n return (\n <div data-slot=\"sidebar-menu-skeleton\" data-sidebar=\"menu-skeleton\" className={cn('flex h-8 items-center gap-2 rounded-md px-2', className)} {...props}>\n {showIcon && <Skeleton className=\"size-4 rounded-md\" data-sidebar=\"menu-skeleton-icon\" />}\n <Skeleton\n className=\"h-4 max-w-(--skeleton-width) flex-1\"\n data-sidebar=\"menu-skeleton-text\"\n style={\n {\n '--skeleton-width': width,\n } as React.CSSProperties\n }\n />\n </div>\n );\n}\n\nfunction ServiceLayoutSidebarMenuSub({ className, ...props }: React.ComponentProps<'ul'>) {\n return (\n <ul\n data-slot=\"sidebar-menu-sub\"\n data-sidebar=\"menu-sub\"\n className={cn(\n 'mx-3.5 flex min-w-0 translate-x-px flex-col gap-1 border-sidebar-border border-l px-2.5 py-0.5',\n 'group-data-[collapsible=icon]:hidden',\n className\n )}\n {...props}\n />\n );\n}\n\nfunction ServiceLayoutSidebarMenuSubItem({ className, ...props }: React.ComponentProps<'li'>) {\n return <li data-slot=\"sidebar-menu-sub-item\" data-sidebar=\"menu-sub-item\" className={cn('group/menu-sub-item relative', className)} {...props} />;\n}\n\nfunction ServiceLayoutSidebarMenuSubButton({\n asChild = false,\n size = 'md',\n isActive = false,\n className,\n ...props\n}: React.ComponentProps<'a'> & {\n asChild?: boolean;\n size?: 'sm' | 'md';\n isActive?: boolean;\n}) {\n const Comp = asChild ? SlotPrimitive.Slot : 'a';\n\n return (\n <Comp\n data-slot=\"sidebar-menu-sub-button\"\n data-sidebar=\"menu-sub-button\"\n data-size={size}\n data-active={isActive}\n className={cn(\n 'text-sidebar-foreground',\n 'ring-sidebar-ring',\n 'flex h-7 min-w-0 -translate-x-px items-center gap-2 overflow-hidden rounded-md px-2 outline-hidden',\n 'focus-visible:ring-2',\n 'hover:bg-sidebar-accent',\n 'hover:text-sidebar-accent-foreground',\n 'active:bg-sidebar-accent',\n 'active:text-sidebar-accent-foreground',\n 'disabled:pointer-events-none',\n 'disabled:opacity-50',\n 'aria-disabled:pointer-events-none',\n 'aria-disabled:opacity-50',\n '[&>svg]:size-4',\n '[&>svg]:shrink-0',\n '[&>svg]:text-sidebar-accent-foreground',\n '[&>span:last-child]:truncate',\n 'data-[active=true]:bg-sidebar-accent',\n 'data-[active=true]:text-sidebar-accent-foreground',\n size === 'sm' && 'text-xs',\n size === 'md' && 'text-sm',\n 'group-data-[collapsible=icon]:hidden',\n className\n )}\n {...props}\n />\n );\n}\n\nexport {\n ServiceLayoutSidebar,\n ServiceLayoutSidebarContent,\n ServiceLayoutSidebarFooter,\n ServiceLayoutSidebarGroup,\n ServiceLayoutSidebarGroupAction,\n ServiceLayoutSidebarGroupContent,\n ServiceLayoutSidebarGroupLabel,\n ServiceLayoutSidebarHeader,\n ServiceLayoutSidebarInset,\n ServiceLayoutSidebarMenu,\n ServiceLayoutSidebarMenuAction,\n ServiceLayoutSidebarMenuBadge,\n ServiceLayoutSidebarMenuButton,\n ServiceLayoutSidebarMenuItem,\n ServiceLayoutSidebarMenuSkeleton,\n ServiceLayoutSidebarMenuSub,\n ServiceLayoutSidebarMenuSubButton,\n ServiceLayoutSidebarMenuSubItem,\n ServiceLayoutSidebarProvider,\n ServiceLayoutSidebarRail,\n ServiceLayoutSidebarSeparator,\n ServiceLayoutSidebarTrigger,\n // biome-ignore lint/style/useComponentExportOnlyModules: true\n useServiceLayoutSidebar,\n};\n","'use client';\nimport { useState } from 'react';\n\nimport { LockIcon, LogInIcon, LogOutIcon, ShoppingBag, ShoppingCartIcon, UserIcon } from 'lucide-react';\n\nimport { useMediaQuery } from '@customafk/react-toolkit/hooks/useMediaQuery';\nimport { cn } from '@customafk/react-toolkit/utils';\n\nimport { Avatar, AvatarFallback, AvatarImage } from '@/components/ui/avatar';\nimport { Button } from '@/components/ui/button';\nimport { Dialog, DialogContent, DialogFooter, DialogHeader, DialogTitle } from '@/components/ui/dialog';\nimport { Drawer, DrawerContent, DrawerFooter, DrawerHeader, DrawerTitle } from '@/components/ui/drawer';\nimport {\n DropdownMenu,\n DropdownMenuContent,\n DropdownMenuItem,\n DropdownMenuLabel,\n DropdownMenuSeparator,\n DropdownMenuTrigger,\n} from '@/components/ui/dropdown-menu';\nimport { Separator } from '@/components/ui/separator';\nimport { Sheet, SheetContent, SheetHeader, SheetTitle, SheetTrigger } from '@/components/ui/sheet';\nimport { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs';\n\nimport { GoogleLogin } from '@react-oauth/google';\nimport { CartList } from './components/cart';\nimport { ServiceLayoutContext, type ServiceLayoutContextProps, useServiceLayout } from './hooks/use-service-layout';\nimport {\n ServiceLayoutSidebar as LayoutSidebar,\n ServiceLayoutSidebarInset,\n ServiceLayoutSidebarProvider,\n ServiceLayoutSidebarTrigger,\n} from './service-layout-sidebar';\n\nexport const ServiceLayoutProvider: React.FC<React.PropsWithChildren<ServiceLayoutContextProps>> = ({\n isLoggedIn = false,\n username,\n email,\n inStockCarts,\n orderedCarts,\n onGoogleLoginSuccess,\n onUpdatingCart,\n onDeletingCart,\n onLogout,\n children,\n}) => {\n return (\n <ServiceLayoutContext.Provider\n value={{\n isLoggedIn,\n username,\n email,\n inStockCarts,\n orderedCarts,\n onGoogleLoginSuccess,\n onUpdatingCart,\n onDeletingCart,\n onLogout,\n }}\n >\n {children}\n </ServiceLayoutContext.Provider>\n );\n};\n\nexport const ServiceLayoutWrapper: React.FC<React.PropsWithChildren> = ({ children }) => {\n return <ServiceLayoutSidebarProvider>{children}</ServiceLayoutSidebarProvider>;\n};\n\ntype ServiceLayoutUserInfoProps = {\n userName?: string;\n userEmail?: string;\n onLogout?: () => void | Promise<void>;\n};\nexport const ServiceLayoutUserInfo: React.FC<ServiceLayoutUserInfoProps> = ({ userName = 'Keith Kennedy', userEmail = 'k.kennedy@originui.com', onLogout }) => {\n return (\n <DropdownMenu>\n <DropdownMenuTrigger asChild>\n <Button size=\"icon\" variant=\"ghost\" color=\"secondary\" className=\"size-10 rounded-full\">\n <Avatar className=\"size-10\">\n <AvatarImage />\n <AvatarFallback className=\"size-full bg-muted-muted\">\n <UserIcon />\n </AvatarFallback>\n </Avatar>\n </Button>\n </DropdownMenuTrigger>\n <DropdownMenuContent align=\"end\" className=\"max-w-64\">\n <DropdownMenuLabel className=\"flex min-w-0 flex-col\">\n <span className=\"truncate font-medium text-sm text-text-positive\">{userName}</span>\n <span className=\"truncate font-normal text-text-positive-weak text-xs\">{userEmail}</span>\n </DropdownMenuLabel>\n <DropdownMenuSeparator />\n <DropdownMenuItem onClick={onLogout}>\n <LogOutIcon size={16} aria-hidden=\"true\" />\n <span className=\"text-text-positive\">Logout</span>\n </DropdownMenuItem>\n </DropdownMenuContent>\n </DropdownMenu>\n );\n};\n\nexport const ServiceLayoutCartInfo: React.FC = () => {\n const { inStockCarts = [], orderedCarts = [] } = useServiceLayout();\n const inStockCount = inStockCarts.length;\n const preOrderCount = orderedCarts.length;\n const totalItems = inStockCount + preOrderCount;\n\n return (\n <Sheet>\n <SheetTrigger asChild>\n <Button size=\"icon\" variant=\"ghost\" color=\"secondary\" className=\"relative size-10 rounded-full\">\n <ShoppingCartIcon />\n </Button>\n </SheetTrigger>\n <SheetContent className=\"w-[95vw] sm:max-w-md\">\n <SheetHeader className=\"flex-0 border-border-weak border-b pb-3\">\n <SheetTitle className=\"flex items-center gap-2\">\n <ShoppingBag size={20} />\n <span>Giỏ hàng của bạn</span>\n {totalItems > 0 && <span className=\"font-normal text-sm text-text-positive-weak\">({totalItems} sản phẩm)</span>}\n </SheetTitle>\n </SheetHeader>\n <Tabs defaultValue={preOrderCount > 0 ? 'pre_order' : 'in_stock'} className=\"h-full flex-1 overflow-y-auto p-4 pt-0\">\n <TabsList className=\"w-full flex-0\">\n <TabsTrigger value=\"in_stock\" className=\"relative\">\n Có sẵn\n {inStockCount > 0 && <span>({inStockCount})</span>}\n </TabsTrigger>\n <TabsTrigger value=\"pre_order\" className=\"relative\">\n Đặt trước\n {preOrderCount > 0 && <span>({preOrderCount})</span>}\n </TabsTrigger>\n </TabsList>\n <TabsContent value=\"pre_order\" className=\"flex-1 overflow-y-auto\">\n <CartList items={orderedCarts} cartType=\"pre_order\" />\n </TabsContent>\n <TabsContent value=\"in_stock\" className=\"flex-1 overflow-y-auto\">\n <CartList items={inStockCarts} cartType=\"in_stock\" />\n </TabsContent>\n </Tabs>\n </SheetContent>\n </Sheet>\n );\n};\n\ntype ServiceLayoutHeaderProps = {\n isLoggedIn?: boolean;\n};\nexport const ServiceLayoutHeader: React.FC<ServiceLayoutHeaderProps> = () => {\n const { isLoggedIn, username, email, onGoogleLoginSuccess, onLogout } = useServiceLayout();\n\n const isDesktop = useMediaQuery('(min-width: 640px)');\n\n const [loginOpen, setLoginOpen] = useState<boolean>(false);\n\n return (\n <header\n className={cn(\n 'bg-card',\n 'h-(--header-height)',\n 'sm:h-[calc(var(--header-height)+0.5rem)] sm:px-4 sm:pr-6',\n 'absolute inset-x-0 top-0 z-20 gap-2 px-2 pr-4.5',\n 'flex items-center shadow-nav',\n 'transition-[width,height] ease-linear'\n )}\n >\n {isLoggedIn && <ServiceLayoutSidebarTrigger />}\n\n <div className=\"flex gap-x-2 sm:ml-2.5\">\n <div className=\"flex aspect-square size-8 items-center justify-center rounded-lg bg-sidebar-primary text-sidebar-primary-foreground\">\n <ShoppingCartIcon size={20} />\n </div>\n <div className=\"grid flex-1 text-left text-sm leading-tight\">\n <span className=\"truncate font-medium\">Lunas Store</span>\n <span className=\"truncate text-xs\">Established 2023</span>\n </div>\n </div>\n\n <div className=\"flex flex-1 items-center justify-end gap-x-2\">\n {!isLoggedIn && (\n <Button className=\"w-8 sm:w-fit\" onClick={() => setLoginOpen(true)}>\n <LogInIcon />\n <span className=\"sr-only sm:not-sr-only\">Đăng nhập</span>\n </Button>\n )}\n {isLoggedIn && (\n <>\n <ServiceLayoutUserInfo userName={username} userEmail={email} onLogout={onLogout} />\n <Separator orientation=\"vertical\" className=\"min-h-6 w-px\" />\n <ServiceLayoutCartInfo />\n </>\n )}\n </div>\n\n {isDesktop && (\n <Dialog open={loginOpen} onOpenChange={setLoginOpen}>\n <DialogContent showCloseButton={false} className=\"flex flex-col gap-0 border-none p-0 sm:max-w-sm\">\n <DialogHeader className=\"flex-0 gap-2 p-6\">\n <DialogTitle className=\"text-center\">Chào mừng bạn đến với Lunas Store!</DialogTitle>\n </DialogHeader>\n <div className=\"flex flex-1 flex-col\">\n <main className=\"size-full flex-1 bg-card p-4 pt-0\">\n <div className=\"flex flex-col items-center gap-y-1\">\n <p className=\"text-sm text-text-positive-weak\">Đăng nhập với Google</p>\n <GoogleLogin\n size=\"large\"\n theme=\"outline\"\n width={240}\n onSuccess={async response => {\n if (!response.clientId || !response.credential || !response.select_by) return;\n await onGoogleLoginSuccess?.(response);\n }}\n />\n </div>\n </main>\n </div>\n <DialogFooter className=\"p-2\" />\n </DialogContent>\n </Dialog>\n )}\n\n {!isDesktop && (\n <Drawer open={loginOpen} onOpenChange={setLoginOpen}>\n <DrawerContent>\n <DrawerHeader className=\"text-left\">\n <DrawerTitle>Chào mừng bạn đến với Lunas Store!</DrawerTitle>\n </DrawerHeader>\n <div className=\"flex flex-1 flex-col\">\n <main className=\"flex size-full flex-1 flex-col p-4 pt-0\">\n <div className=\"flex flex-col items-center gap-y-1\">\n <p className=\"text-sm text-text-positive-weak\">Đăng nhập với Google</p>\n <GoogleLogin\n size=\"large\"\n theme=\"outline\"\n onSuccess={async response => {\n if (!response.clientId || !response.credential || !response.select_by) return;\n await onGoogleLoginSuccess?.(response);\n }}\n />\n </div>\n </main>\n </div>\n <DrawerFooter />\n </DrawerContent>\n </Drawer>\n )}\n </header>\n );\n};\n\nexport const ServiceLayoutSidebar: React.FC<React.PropsWithChildren & React.ComponentProps<typeof LayoutSidebar>> = ({ children, ...props }) => {\n const { isLoggedIn } = useServiceLayout();\n if (!isLoggedIn) return null;\n return (\n <LayoutSidebar variant=\"inset\" collapsible=\"icon\" {...props}>\n {children}\n </LayoutSidebar>\n );\n};\n\nexport const ServiceLayoutMain: React.FC<React.PropsWithChildren> = ({ children }) => {\n const { isLoggedIn, onGoogleLoginSuccess } = useServiceLayout();\n const isDesktop = useMediaQuery('(min-width: 640px)');\n const [loginOpen, setLoginOpen] = useState<boolean>(false);\n\n if (!isLoggedIn) {\n return (\n <div className=\"size-full p-4 pt-[calc(var(--header-height)+1.5rem)]\">\n <div className=\"flex size-full flex-col items-center justify-center gap-6 rounded-lg bg-card p-8 text-center shadow-card\">\n <div className=\"flex size-20 items-center justify-center rounded-full bg-muted-foreground/10\">\n <LockIcon className=\"size-10 text-primary\" />\n </div>\n <div className=\"flex max-w-md flex-col gap-2\">\n <h2 className=\"font-semibold text-2xl\">Bạn chưa đăng nhập</h2>\n <p className=\"text-text-positive-weak\">Đăng nhập để khám phá đầy đủ các tính năng của Lunas Store và truy cập vào tài khoản của bạn.</p>\n </div>\n <Button size=\"lg\" className=\"gap-2\" onClick={() => setLoginOpen(true)}>\n <LogInIcon size={18} />\n <span>Đăng nhập ngay</span>\n </Button>\n\n {isDesktop && (\n <Dialog open={loginOpen} onOpenChange={setLoginOpen}>\n <DialogContent showCloseButton={false} className=\"flex flex-col gap-0 border-none p-0 sm:max-w-sm\">\n <DialogHeader className=\"flex-0 gap-2 p-6\">\n <DialogTitle className=\"text-center\">Chào mừng bạn đến với Lunas Store!</DialogTitle>\n </DialogHeader>\n <div className=\"flex flex-1 flex-col\">\n <main className=\"size-full flex-1 bg-card p-4 pt-0\">\n <div className=\"flex flex-col items-center gap-y-1\">\n <p className=\"text-sm text-text-positive-weak\">Đăng nhập với Google</p>\n <GoogleLogin\n size=\"large\"\n theme=\"outline\"\n width={240}\n onSuccess={async response => {\n if (!response.clientId || !response.credential || !response.select_by) return;\n await onGoogleLoginSuccess?.(response);\n setLoginOpen(false);\n }}\n />\n </div>\n </main>\n </div>\n <DialogFooter className=\"p-2\" />\n </DialogContent>\n </Dialog>\n )}\n\n {!isDesktop && (\n <Drawer open={loginOpen} onOpenChange={setLoginOpen}>\n <DrawerContent>\n <DrawerHeader className=\"text-left\">\n <DrawerTitle>Chào mừng bạn đến với Lunas Store!</DrawerTitle>\n </DrawerHeader>\n <div className=\"flex flex-1 flex-col\">\n <main className=\"flex size-full flex-1 flex-col p-4 pt-0\">\n <div className=\"flex flex-col items-center gap-y-1\">\n <p className=\"text-sm text-text-positive-weak\">Đăng nhập với Google</p>\n <GoogleLogin\n size=\"large\"\n theme=\"outline\"\n onSuccess={async response => {\n if (!response.clientId || !response.credential || !response.select_by) return;\n await onGoogleLoginSuccess?.(response);\n setLoginOpen(false);\n }}\n />\n </div>\n </main>\n </div>\n <DrawerFooter />\n </DrawerContent>\n </Drawer>\n )}\n </div>\n </div>\n );\n }\n\n return (\n <ServiceLayoutSidebarInset>\n <section className=\"relative size-full\">\n <div className=\"absolute inset-0 flex flex-col\">{children}</div>\n </section>\n </ServiceLayoutSidebarInset>\n );\n};\nexport const ServiceLayoutMainHeader: React.FC<React.PropsWithChildren & { className?: string }> = ({ className, children }) => {\n return (\n <div data-slot=\"main-header\" className={cn('flex-0 snap-start', className)}>\n {children}\n </div>\n );\n};\n\nexport const ServiceLayoutMainContent: React.FC<React.PropsWithChildren & { className?: string }> = ({ className, children }) => {\n return (\n <div data-slot=\"main-content\" className={cn('flex w-full flex-1 flex-col gap-4 overflow-y-auto px-2 sm:px-4', className)}>\n {children}\n </div>\n );\n};\n\nexport const ServiceLayoutMainFooter: React.FC<React.PropsWithChildren & { className?: string }> = ({ className, children }) => {\n return (\n <div data-slot=\"main-footer\" className={cn('hidden w-full flex-0 border-border-weak border-t pt-2 sm:flex', className)}>\n {children}\n </div>\n );\n};\n\nexport const ServiceLayoutMainGroup: React.FC<React.PropsWithChildren<{ className?: string }>> = ({ children, className }) => {\n return (\n <div data-slot=\"main-group\" className={cn('flex size-full flex-col gap-4', className)}>\n {children}\n </div>\n );\n};\n\nexport const ServiceLayoutMainGroupContent: React.FC<React.PropsWithChildren & { className?: string }> = ({ className, children }) => {\n return (\n <div data-slot=\"main-group-content\" className={cn('size-full max-w-8xl flex-1 rounded-md bg-card p-4 shadow-card', className)}>\n {children}\n </div>\n );\n};\n"],"mappings":"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"}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import * as
|
|
1
|
+
import * as react_jsx_runtime7 from "react/jsx-runtime";
|
|
2
2
|
|
|
3
3
|
//#region packages/components/pages/FeatureDeveloping.d.ts
|
|
4
4
|
interface FeatureDevelopingProps {
|
|
@@ -19,7 +19,7 @@ declare const FeatureDeveloping: ({
|
|
|
19
19
|
onButtonClick,
|
|
20
20
|
className,
|
|
21
21
|
iconClassName
|
|
22
|
-
}: FeatureDevelopingProps) =>
|
|
22
|
+
}: FeatureDevelopingProps) => react_jsx_runtime7.JSX.Element;
|
|
23
23
|
//#endregion
|
|
24
24
|
export { FeatureDeveloping, FeatureDeveloping as default, FeatureDevelopingProps };
|
|
25
25
|
//# sourceMappingURL=FeatureDeveloping.d.cts.map
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import * as
|
|
1
|
+
import * as react_jsx_runtime26 from "react/jsx-runtime";
|
|
2
2
|
|
|
3
3
|
//#region packages/components/pages/FeatureDeveloping.d.ts
|
|
4
4
|
interface FeatureDevelopingProps {
|
|
@@ -19,7 +19,7 @@ declare const FeatureDeveloping: ({
|
|
|
19
19
|
onButtonClick,
|
|
20
20
|
className,
|
|
21
21
|
iconClassName
|
|
22
|
-
}: FeatureDevelopingProps) =>
|
|
22
|
+
}: FeatureDevelopingProps) => react_jsx_runtime26.JSX.Element;
|
|
23
23
|
//#endregion
|
|
24
24
|
export { FeatureDeveloping, FeatureDeveloping as default, FeatureDevelopingProps };
|
|
25
25
|
//# sourceMappingURL=FeatureDeveloping.d.mts.map
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import"../button.variants-
|
|
1
|
+
import"../button.variants-DmUzMbB8.mjs";import{t as e}from"../button-DIEXNBpi.mjs";import{cn as t}from"@customafk/react-toolkit/utils";import{jsx as n,jsxs as r}from"react/jsx-runtime";import{Construction as i}from"lucide-react";const a=({title:a=`Tính năng đang phát triển`,subtitle:o=`Chúng tôi đang phát triển tính năng này. Vui lòng quay lại sau.`,buttonText:s=`Quay lại`,onButtonClick:c=()=>window.history.back(),className:l,iconClassName:u})=>r(`div`,{className:t(`flex flex-col items-center justify-center min-h-[50vh] gap-4 sm:gap-6 py-8 sm:py-10 px-4 text-center w-full`,l),children:[r(`div`,{className:`flex flex-col items-center gap-2`,children:[n(i,{className:t(`size-16 sm:size-24 text-yellow-500`,u)}),n(`div`,{className:`text-3xl sm:text-5xl font-bold text-yellow-500`})]}),n(`h1`,{className:`text-xl sm:text-2xl font-semibold mt-2`,children:a}),n(`p`,{className:`text-muted-foreground max-w-md text-sm sm:text-base`,children:o}),n(e,{onClick:c,variant:`default`,className:`mt-2`,size:`sm`,children:s})]});var o=a;export{a as FeatureDeveloping,o as default};
|
|
2
2
|
//# sourceMappingURL=FeatureDeveloping.mjs.map
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import * as
|
|
1
|
+
import * as react_jsx_runtime4 from "react/jsx-runtime";
|
|
2
2
|
|
|
3
3
|
//#region packages/components/pages/FeatureFixing.d.ts
|
|
4
4
|
interface FeatureFixingProps {
|
|
@@ -19,7 +19,7 @@ declare const FeatureFixing: ({
|
|
|
19
19
|
onButtonClick,
|
|
20
20
|
className,
|
|
21
21
|
iconClassName
|
|
22
|
-
}: FeatureFixingProps) =>
|
|
22
|
+
}: FeatureFixingProps) => react_jsx_runtime4.JSX.Element;
|
|
23
23
|
//#endregion
|
|
24
24
|
export { FeatureFixing, FeatureFixing as default, FeatureFixingProps };
|
|
25
25
|
//# sourceMappingURL=FeatureFixing.d.cts.map
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import * as
|
|
1
|
+
import * as react_jsx_runtime29 from "react/jsx-runtime";
|
|
2
2
|
|
|
3
3
|
//#region packages/components/pages/FeatureFixing.d.ts
|
|
4
4
|
interface FeatureFixingProps {
|
|
@@ -19,7 +19,7 @@ declare const FeatureFixing: ({
|
|
|
19
19
|
onButtonClick,
|
|
20
20
|
className,
|
|
21
21
|
iconClassName
|
|
22
|
-
}: FeatureFixingProps) =>
|
|
22
|
+
}: FeatureFixingProps) => react_jsx_runtime29.JSX.Element;
|
|
23
23
|
//#endregion
|
|
24
24
|
export { FeatureFixing, FeatureFixing as default, FeatureFixingProps };
|
|
25
25
|
//# sourceMappingURL=FeatureFixing.d.mts.map
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import"../button.variants-
|
|
1
|
+
import"../button.variants-DmUzMbB8.mjs";import{t as e}from"../button-DIEXNBpi.mjs";import{cn as t}from"@customafk/react-toolkit/utils";import{jsx as n,jsxs as r}from"react/jsx-runtime";import{Wrench as i}from"lucide-react";const a=({title:a=`Tính năng đang bảo trì`,subtitle:o=`Chúng tôi đang khắc phục vấn đề với tính năng này. Vui lòng quay lại sau.`,buttonText:s=`Quay lại`,onButtonClick:c=()=>window.history.back(),className:l,iconClassName:u})=>r(`div`,{className:t(`flex flex-col items-center justify-center min-h-[50vh] gap-4 sm:gap-6 py-8 sm:py-10 px-4 text-center w-full`,l),children:[r(`div`,{className:`flex flex-col items-center gap-2`,children:[n(i,{className:t(`size-16 sm:size-24 text-orange-500`,u)}),n(`div`,{className:`text-3xl sm:text-5xl font-bold text-orange-500`})]}),n(`h1`,{className:`text-xl sm:text-2xl font-semibold mt-2`,children:a}),n(`p`,{className:`text-muted-foreground max-w-md text-sm sm:text-base`,children:o}),n(e,{onClick:c,variant:`default`,className:`mt-2`,size:`sm`,children:s})]});var o=a;export{a as FeatureFixing,o as default};
|
|
2
2
|
//# sourceMappingURL=FeatureFixing.mjs.map
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import * as
|
|
1
|
+
import * as react_jsx_runtime11 from "react/jsx-runtime";
|
|
2
2
|
|
|
3
3
|
//#region packages/components/pages/NotAuthorized.d.ts
|
|
4
4
|
interface NotAuthorizedProps {
|
|
@@ -19,7 +19,7 @@ declare const NotAuthorized: ({
|
|
|
19
19
|
onButtonClick,
|
|
20
20
|
className,
|
|
21
21
|
iconClassName
|
|
22
|
-
}: NotAuthorizedProps) =>
|
|
22
|
+
}: NotAuthorizedProps) => react_jsx_runtime11.JSX.Element;
|
|
23
23
|
//#endregion
|
|
24
24
|
export { NotAuthorized, NotAuthorized as default, NotAuthorizedProps };
|
|
25
25
|
//# sourceMappingURL=NotAuthorized.d.cts.map
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import * as
|
|
1
|
+
import * as react_jsx_runtime31 from "react/jsx-runtime";
|
|
2
2
|
|
|
3
3
|
//#region packages/components/pages/NotAuthorized.d.ts
|
|
4
4
|
interface NotAuthorizedProps {
|
|
@@ -19,7 +19,7 @@ declare const NotAuthorized: ({
|
|
|
19
19
|
onButtonClick,
|
|
20
20
|
className,
|
|
21
21
|
iconClassName
|
|
22
|
-
}: NotAuthorizedProps) =>
|
|
22
|
+
}: NotAuthorizedProps) => react_jsx_runtime31.JSX.Element;
|
|
23
23
|
//#endregion
|
|
24
24
|
export { NotAuthorized, NotAuthorized as default, NotAuthorizedProps };
|
|
25
25
|
//# sourceMappingURL=NotAuthorized.d.mts.map
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import"../button.variants-
|
|
1
|
+
import"../button.variants-DmUzMbB8.mjs";import{t as e}from"../button-DIEXNBpi.mjs";import{cn as t}from"@customafk/react-toolkit/utils";import{jsx as n,jsxs as r}from"react/jsx-runtime";import{ShieldAlert as i}from"lucide-react";const a=({title:a=`Quyền truy cập bị từ chối`,subtitle:o=`Xin lỗi, bạn không có quyền truy cập vào trang này.`,buttonText:s=`Quay lại`,onButtonClick:c=()=>window.history.back(),className:l,iconClassName:u})=>r(`div`,{className:t(`flex flex-col items-center justify-center min-h-[50vh] gap-4 sm:gap-6 py-8 sm:py-10 px-4 text-center w-full`,l),children:[r(`div`,{className:`flex flex-col items-center gap-2`,children:[n(i,{className:t(`size-16 sm:size-24 text-primary`,u)}),n(`div`,{className:`text-4xl sm:text-6xl font-bold text-primary`,children:`403`})]}),n(`h1`,{className:`text-xl sm:text-2xl font-semibold mt-2`,children:a}),n(`p`,{className:`text-muted-foreground max-w-md text-sm sm:text-base`,children:o}),n(e,{onClick:c,variant:`default`,className:`mt-2`,size:`sm`,children:s})]});var o=a;export{a as NotAuthorized,o as default};
|
|
2
2
|
//# sourceMappingURL=NotAuthorized.mjs.map
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import * as
|
|
1
|
+
import * as react_jsx_runtime25 from "react/jsx-runtime";
|
|
2
2
|
|
|
3
3
|
//#region packages/components/pages/NotFound.d.ts
|
|
4
4
|
interface NotFoundProps {
|
|
@@ -19,7 +19,7 @@ declare const NotFound: ({
|
|
|
19
19
|
onButtonClick,
|
|
20
20
|
className,
|
|
21
21
|
iconClassName
|
|
22
|
-
}: NotFoundProps) =>
|
|
22
|
+
}: NotFoundProps) => react_jsx_runtime25.JSX.Element;
|
|
23
23
|
//#endregion
|
|
24
24
|
export { NotFound, NotFound as default, NotFoundProps };
|
|
25
25
|
//# sourceMappingURL=NotFound.d.cts.map
|