@customafk/lunas-ui 0.0.76 → 0.0.78

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (355) hide show
  1. package/dist/{add-new-_5opEbrI.js → add-new-CdyYFK4x.js} +2 -2
  2. package/dist/{add-new-_5opEbrI.js.map → add-new-CdyYFK4x.js.map} +1 -1
  3. package/dist/{add-new-BaA0yy6U.cjs → add-new-CnZV8qWt.cjs} +2 -2
  4. package/dist/{add-new-BaA0yy6U.cjs.map → add-new-CnZV8qWt.cjs.map} +1 -1
  5. package/dist/{alert-dialog-BgEaBYoh.cjs → alert-dialog-DIO7g0Aa.cjs} +1 -1
  6. package/dist/{alert-dialog-BgEaBYoh.cjs.map → alert-dialog-DIO7g0Aa.cjs.map} +1 -1
  7. package/dist/{alert-dialog-UwGNuTYk.js → alert-dialog-lXAg3Alc.js} +1 -1
  8. package/dist/{alert-dialog-UwGNuTYk.js.map → alert-dialog-lXAg3Alc.js.map} +1 -1
  9. package/dist/{avatar-iX5Jhj8E.js → avatar-BT0PH1Tk.js} +1 -1
  10. package/dist/{avatar-iX5Jhj8E.js.map → avatar-BT0PH1Tk.js.map} +1 -1
  11. package/dist/{avatar-B4pklWJt.cjs → avatar-kxLA5n-m.cjs} +1 -1
  12. package/dist/{avatar-B4pklWJt.cjs.map → avatar-kxLA5n-m.cjs.map} +1 -1
  13. package/dist/{badge-CsA-JCDX.cjs → badge-DG8ELRGF.cjs} +1 -1
  14. package/dist/{badge-CsA-JCDX.cjs.map → badge-DG8ELRGF.cjs.map} +1 -1
  15. package/dist/{badge-tjlLdDGl.js → badge-DH9VoXhk.js} +1 -1
  16. package/dist/{badge-tjlLdDGl.js.map → badge-DH9VoXhk.js.map} +1 -1
  17. package/dist/{button-F88fcpt8.d.ts → button-Bpoqzxk-.d.cts} +5 -5
  18. package/dist/{button-ewMXluO3.d.cts → button-BwKsSigg.d.ts} +6 -6
  19. package/dist/{calendar-Fjm1d0mU.cjs → calendar-DGqBIV7n.cjs} +1 -1
  20. package/dist/{calendar-Fjm1d0mU.cjs.map → calendar-DGqBIV7n.cjs.map} +1 -1
  21. package/dist/{calendar-C2yGJ9AV.js → calendar-DdIUj5Dk.js} +1 -1
  22. package/dist/{calendar-C2yGJ9AV.js.map → calendar-DdIUj5Dk.js.map} +1 -1
  23. package/dist/cards/simple-card.d.cts +2 -2
  24. package/dist/{command-DHGZlBKd.cjs → command-BFIF8llT.cjs} +2 -2
  25. package/dist/{command-DHGZlBKd.cjs.map → command-BFIF8llT.cjs.map} +1 -1
  26. package/dist/{command-BwGqftX8.d.ts → command-BUnu6Cro.d.ts} +12 -12
  27. package/dist/{command-DhSE6UaR.js → command-Bdz8IRdV.js} +2 -2
  28. package/dist/{command-DhSE6UaR.js.map → command-Bdz8IRdV.js.map} +1 -1
  29. package/dist/{command-T3juozDI.d.cts → command-C-4VFIwT.d.cts} +12 -12
  30. package/dist/data-display/country.cjs +1 -1
  31. package/dist/data-display/country.js +1 -1
  32. package/dist/data-display/data-list.cjs +1 -1
  33. package/dist/data-display/data-list.js +1 -1
  34. package/dist/data-display/date-tooltip.cjs +1 -1
  35. package/dist/data-display/date-tooltip.js +1 -1
  36. package/dist/data-display/date.cjs +1 -1
  37. package/dist/data-display/date.js +1 -1
  38. package/dist/data-display/empty.cjs +1 -1
  39. package/dist/data-display/empty.d.ts +2 -2
  40. package/dist/data-display/empty.js +1 -1
  41. package/dist/data-display/name.cjs +1 -1
  42. package/dist/data-display/name.js +1 -1
  43. package/dist/data-display/phone-number.cjs +1 -1
  44. package/dist/data-display/phone-number.js +1 -1
  45. package/dist/data-display/role-badge.cjs +1 -1
  46. package/dist/data-display/role-badge.js +1 -1
  47. package/dist/data-display/statistic.cjs +1 -1
  48. package/dist/data-display/statistic.d.cts +2 -2
  49. package/dist/data-display/statistic.d.ts +2 -2
  50. package/dist/data-display/statistic.js +1 -1
  51. package/dist/data-display/user.cjs +1 -1
  52. package/dist/data-display/user.js +1 -1
  53. package/dist/{date-BuPl--Fc.js → date-CAkONFNM.js} +1 -1
  54. package/dist/{date-BuPl--Fc.js.map → date-CAkONFNM.js.map} +1 -1
  55. package/dist/{date-2wue8L-1.cjs → date-Cckz-1ou.cjs} +1 -1
  56. package/dist/{date-2wue8L-1.cjs.map → date-Cckz-1ou.cjs.map} +1 -1
  57. package/dist/{dialog-BSdn3QaW.cjs → dialog-B0nEcw96.cjs} +1 -1
  58. package/dist/{dialog-BSdn3QaW.cjs.map → dialog-B0nEcw96.cjs.map} +1 -1
  59. package/dist/{dialog-Dn2Q-QRw.d.ts → dialog-B4XWQE0p.d.cts} +12 -12
  60. package/dist/{dialog-B2LS2JLb.js → dialog-CjX94zKb.js} +1 -1
  61. package/dist/{dialog-B2LS2JLb.js.map → dialog-CjX94zKb.js.map} +1 -1
  62. package/dist/{dialog-Bw_T5aco.d.cts → dialog-DDsbgE3Z.d.ts} +12 -12
  63. package/dist/dialogs/confirm-dialog.cjs +1 -1
  64. package/dist/dialogs/confirm-dialog.js +1 -1
  65. package/dist/dialogs/detail-dialog/component/sidebar.cjs +1 -1
  66. package/dist/dialogs/detail-dialog/component/sidebar.d.cts +30 -30
  67. package/dist/dialogs/detail-dialog/component/sidebar.d.ts +4 -4
  68. package/dist/dialogs/detail-dialog/component/sidebar.js +1 -1
  69. package/dist/dialogs/detail-dialog/index.cjs +1 -1
  70. package/dist/dialogs/detail-dialog/index.cjs.map +1 -1
  71. package/dist/dialogs/detail-dialog/index.js +1 -1
  72. package/dist/dialogs/error-dialog.cjs +1 -1
  73. package/dist/dialogs/error-dialog.js +1 -1
  74. package/dist/dialogs/form-dialog.cjs +1 -1
  75. package/dist/dialogs/form-dialog.d.cts +2 -2
  76. package/dist/dialogs/form-dialog.d.ts +2 -2
  77. package/dist/dialogs/form-dialog.js +1 -1
  78. package/dist/dialogs/loading-dialog.cjs +1 -1
  79. package/dist/dialogs/loading-dialog.js +1 -1
  80. package/dist/{drawer-CsCzBHvx.cjs → drawer-BWmBT8Xx.cjs} +1 -1
  81. package/dist/{drawer-CsCzBHvx.cjs.map → drawer-BWmBT8Xx.cjs.map} +1 -1
  82. package/dist/{drawer-COD1n5aA.js → drawer-CWQ5bBHi.js} +1 -1
  83. package/dist/{drawer-COD1n5aA.js.map → drawer-CWQ5bBHi.js.map} +1 -1
  84. package/dist/{error-dialog-C3b4fqAI.cjs → error-dialog-Br3cqI6T.cjs} +2 -2
  85. package/dist/{error-dialog-C3b4fqAI.cjs.map → error-dialog-Br3cqI6T.cjs.map} +1 -1
  86. package/dist/{error-dialog-BF1VNZXk.js → error-dialog-CUUXPd_g.js} +2 -2
  87. package/dist/{error-dialog-BF1VNZXk.js.map → error-dialog-CUUXPd_g.js.map} +1 -1
  88. package/dist/{flex-DS4mxjtf.cjs → flex-3xY2da62.cjs} +1 -1
  89. package/dist/{flex-DS4mxjtf.cjs.map → flex-3xY2da62.cjs.map} +1 -1
  90. package/dist/{flex-D6XrXRQW.js → flex-B_qDiQBK.js} +1 -1
  91. package/dist/{flex-D6XrXRQW.js.map → flex-B_qDiQBK.js.map} +1 -1
  92. package/dist/{form-DyH5iWq7.cjs → form-LlBR5CH6.cjs} +2 -2
  93. package/dist/{form-DyH5iWq7.cjs.map → form-LlBR5CH6.cjs.map} +1 -1
  94. package/dist/{form-BT1TbgK6.js → form-nFp50GQa.js} +2 -2
  95. package/dist/{form-BT1TbgK6.js.map → form-nFp50GQa.js.map} +1 -1
  96. package/dist/{form-wrapper-DaguHv9W.js → form-wrapper-C-4ssC7K.js} +2 -2
  97. package/dist/{form-wrapper-DaguHv9W.js.map → form-wrapper-C-4ssC7K.js.map} +1 -1
  98. package/dist/{form-wrapper-CI2iMUNJ.cjs → form-wrapper-C2nxYJsQ.cjs} +2 -2
  99. package/dist/{form-wrapper-CI2iMUNJ.cjs.map → form-wrapper-C2nxYJsQ.cjs.map} +1 -1
  100. package/dist/forms/combobox-field.cjs +1 -1
  101. package/dist/forms/combobox-field.d.cts +2 -2
  102. package/dist/forms/combobox-field.d.ts +2 -2
  103. package/dist/forms/combobox-field.js +1 -1
  104. package/dist/forms/date-field.cjs +1 -1
  105. package/dist/forms/date-field.d.cts +2 -2
  106. package/dist/forms/date-field.d.ts +2 -2
  107. package/dist/forms/date-field.js +1 -1
  108. package/dist/forms/form-wrapper.cjs +1 -1
  109. package/dist/forms/form-wrapper.d.cts +2 -2
  110. package/dist/forms/form-wrapper.d.ts +2 -2
  111. package/dist/forms/form-wrapper.js +1 -1
  112. package/dist/forms/multi-select-field.cjs +1 -1
  113. package/dist/forms/multi-select-field.cjs.map +1 -1
  114. package/dist/forms/multi-select-field.d.cts +2 -2
  115. package/dist/forms/multi-select-field.d.ts +2 -2
  116. package/dist/forms/multi-select-field.js +1 -1
  117. package/dist/forms/number-field.cjs +1 -1
  118. package/dist/forms/number-field.d.cts +2 -2
  119. package/dist/forms/number-field.d.ts +2 -2
  120. package/dist/forms/number-field.js +1 -1
  121. package/dist/forms/password-field.cjs +1 -1
  122. package/dist/forms/password-field.d.cts +2 -2
  123. package/dist/forms/password-field.d.ts +2 -2
  124. package/dist/forms/password-field.js +1 -1
  125. package/dist/forms/select-field.cjs +1 -1
  126. package/dist/forms/select-field.d.cts +2 -2
  127. package/dist/forms/select-field.d.ts +2 -2
  128. package/dist/forms/select-field.js +1 -1
  129. package/dist/forms/switch-field.cjs +1 -1
  130. package/dist/forms/switch-field.d.cts +2 -2
  131. package/dist/forms/switch-field.d.ts +2 -2
  132. package/dist/forms/switch-field.js +1 -1
  133. package/dist/forms/text-field.cjs +1 -1
  134. package/dist/forms/text-field.d.cts +2 -2
  135. package/dist/forms/text-field.js +1 -1
  136. package/dist/forms/textarea-field.cjs +1 -1
  137. package/dist/forms/textarea-field.d.cts +2 -2
  138. package/dist/forms/textarea-field.d.ts +2 -2
  139. package/dist/forms/textarea-field.js +1 -1
  140. package/dist/{input-CwuYXUPY.d.ts → input-8EIWjE5j.d.ts} +3 -3
  141. package/dist/{input-B_LC16rD.d.cts → input-DXE-qRch.d.cts} +3 -3
  142. package/dist/{input-DfGojpRF.cjs → input-GHXBt6hz.cjs} +1 -1
  143. package/dist/{input-DfGojpRF.cjs.map → input-GHXBt6hz.cjs.map} +1 -1
  144. package/dist/{input-BtT8HfYE.js → input-ZyZ1HKHN.js} +1 -1
  145. package/dist/{input-BtT8HfYE.js.map → input-ZyZ1HKHN.js.map} +1 -1
  146. package/dist/{label-CHNSlaoz.cjs → label-BR3QQSIp.cjs} +1 -1
  147. package/dist/{label-CHNSlaoz.cjs.map → label-BR3QQSIp.cjs.map} +1 -1
  148. package/dist/{label-CdrxlBJ8.js → label-CRhvNQY4.js} +1 -1
  149. package/dist/{label-CdrxlBJ8.js.map → label-CRhvNQY4.js.map} +1 -1
  150. package/dist/layouts/app-layout/index.cjs +1 -1
  151. package/dist/layouts/app-layout/index.d.cts +3 -3
  152. package/dist/layouts/app-layout/index.d.ts +28 -28
  153. package/dist/layouts/app-layout/index.js +1 -1
  154. package/dist/layouts/flex.cjs +1 -1
  155. package/dist/layouts/flex.d.cts +2 -2
  156. package/dist/layouts/flex.d.ts +2 -2
  157. package/dist/layouts/flex.js +1 -1
  158. package/dist/layouts/service-layout/index.cjs +1 -1
  159. package/dist/layouts/service-layout/index.d.cts +3 -3
  160. package/dist/layouts/service-layout/index.d.ts +15 -15
  161. package/dist/layouts/service-layout/index.js +1 -1
  162. package/dist/layouts/service-layout/index.js.map +1 -1
  163. package/dist/{multi-select-l0HITJuv.js → multi-select-7gJ0p2qk.js} +2 -2
  164. package/dist/{multi-select-l0HITJuv.js.map → multi-select-7gJ0p2qk.js.map} +1 -1
  165. package/dist/{multi-select-C0dFenI-.cjs → multi-select-zRSNabHi.cjs} +2 -2
  166. package/dist/{multi-select-C0dFenI-.cjs.map → multi-select-zRSNabHi.cjs.map} +1 -1
  167. package/dist/{paragraph-BLvW7xdo.cjs → paragraph-BVNragDZ.cjs} +1 -1
  168. package/dist/{paragraph-BLvW7xdo.cjs.map → paragraph-BVNragDZ.cjs.map} +1 -1
  169. package/dist/{paragraph-B9fp9JaI.js → paragraph-eTkdWYsK.js} +1 -1
  170. package/dist/{paragraph-B9fp9JaI.js.map → paragraph-eTkdWYsK.js.map} +1 -1
  171. package/dist/{popover-BrDe5Jb8.cjs → popover-DLbTz02b.cjs} +1 -1
  172. package/dist/{popover-BrDe5Jb8.cjs.map → popover-DLbTz02b.cjs.map} +1 -1
  173. package/dist/{popover-BJWOgxQ5.js → popover-DvXi2pyH.js} +1 -1
  174. package/dist/{popover-BJWOgxQ5.js.map → popover-DvXi2pyH.js.map} +1 -1
  175. package/dist/{refresh-Bt9GEyuC.cjs → refresh-BMBMI6At.cjs} +2 -2
  176. package/dist/{refresh-Bt9GEyuC.cjs.map → refresh-BMBMI6At.cjs.map} +1 -1
  177. package/dist/{refresh-BHjc0ETv.js → refresh-CWyM3e6G.js} +2 -2
  178. package/dist/{refresh-BHjc0ETv.js.map → refresh-CWyM3e6G.js.map} +1 -1
  179. package/dist/{search-input-CjYhM9sw.cjs → search-input-DFgHJyOp.cjs} +2 -2
  180. package/dist/{search-input-CjYhM9sw.cjs.map → search-input-DFgHJyOp.cjs.map} +1 -1
  181. package/dist/{search-input-CvMbwPyf.js → search-input-DM9ePMxA.js} +2 -2
  182. package/dist/{search-input-CvMbwPyf.js.map → search-input-DM9ePMxA.js.map} +1 -1
  183. package/dist/{select-Di8fOKVF.js → select---jN6Yy0.js} +1 -1
  184. package/dist/{select-Di8fOKVF.js.map → select---jN6Yy0.js.map} +1 -1
  185. package/dist/{select-Ce-YROpe.cjs → select-CvPIAeWb.cjs} +1 -1
  186. package/dist/{select-Ce-YROpe.cjs.map → select-CvPIAeWb.cjs.map} +1 -1
  187. package/dist/{separator-DUeUh8bV.d.ts → separator-DExlvlJ1.d.ts} +3 -3
  188. package/dist/{separator-BSmwGEtT.d.cts → separator-E2_sfBcf.d.cts} +3 -3
  189. package/dist/{sidebar-CBEPWmnf.js → sidebar-Baw-T7Q1.js} +2 -2
  190. package/dist/{sidebar-CBEPWmnf.js.map → sidebar-Baw-T7Q1.js.map} +1 -1
  191. package/dist/{sidebar-B6s0ZZHA.cjs → sidebar-V5_9Ws0A.cjs} +2 -2
  192. package/dist/{sidebar-B6s0ZZHA.cjs.map → sidebar-V5_9Ws0A.cjs.map} +1 -1
  193. package/dist/{switch-DOgkZVAD.cjs → switch-BLfDkOOl.cjs} +1 -1
  194. package/dist/{switch-DOgkZVAD.cjs.map → switch-BLfDkOOl.cjs.map} +1 -1
  195. package/dist/{switch-C7foFKpS.js → switch-C8OMYLfj.js} +1 -1
  196. package/dist/{switch-C7foFKpS.js.map → switch-C8OMYLfj.js.map} +1 -1
  197. package/dist/table/index.cjs +1 -1
  198. package/dist/table/index.d.cts +2 -2
  199. package/dist/table/index.d.ts +2 -2
  200. package/dist/table/index.js +1 -1
  201. package/dist/{textarea-n8OO0QsI.js → textarea-Br_DX1wm.js} +1 -1
  202. package/dist/{textarea-n8OO0QsI.js.map → textarea-Br_DX1wm.js.map} +1 -1
  203. package/dist/{textarea-Ewj89Eq3.cjs → textarea-jrCmozqt.cjs} +1 -1
  204. package/dist/{textarea-Ewj89Eq3.cjs.map → textarea-jrCmozqt.cjs.map} +1 -1
  205. package/dist/{toggle-DTLzwRZc.d.cts → toggle-B4TinVZu.d.cts} +5 -5
  206. package/dist/{toggle-D6-qXZDi.d.ts → toggle-DPTA8XO2.d.ts} +5 -5
  207. package/dist/{tooltip-Bj2uhIpz.d.ts → tooltip-BKTq7o5G.d.ts} +6 -6
  208. package/dist/{tooltip-BTrjCwd8.js → tooltip-CNdZZO_B.js} +1 -1
  209. package/dist/{tooltip-BTrjCwd8.js.map → tooltip-CNdZZO_B.js.map} +1 -1
  210. package/dist/{tooltip-DXvcPsvM.cjs → tooltip-DCbmI7Af.cjs} +1 -1
  211. package/dist/{tooltip-DXvcPsvM.cjs.map → tooltip-DCbmI7Af.cjs.map} +1 -1
  212. package/dist/{tooltip-rj9mqdk5.d.cts → tooltip-DLNDOfBF.d.cts} +6 -6
  213. package/dist/{types-D32qXHff.cjs → types-Cs3x9-zN.cjs} +1 -1
  214. package/dist/{types-D32qXHff.cjs.map → types-Cs3x9-zN.cjs.map} +1 -1
  215. package/dist/{types-BiiGENwa.js → types-DkOgOlMw.js} +1 -1
  216. package/dist/{types-BiiGENwa.js.map → types-DkOgOlMw.js.map} +1 -1
  217. package/dist/typography/paragraph.cjs +1 -1
  218. package/dist/typography/paragraph.d.cts +2 -2
  219. package/dist/typography/paragraph.d.ts +2 -2
  220. package/dist/typography/paragraph.js +1 -1
  221. package/dist/typography/title.d.cts +2 -2
  222. package/dist/typography/title.d.ts +2 -2
  223. package/dist/ui/alert-dialog.cjs +1 -1
  224. package/dist/ui/alert-dialog.d.cts +12 -12
  225. package/dist/ui/alert-dialog.d.ts +12 -12
  226. package/dist/ui/alert-dialog.js +1 -1
  227. package/dist/ui/alert.d.cts +6 -6
  228. package/dist/ui/alert.d.ts +4 -4
  229. package/dist/ui/aspect-ratio.d.cts +2 -2
  230. package/dist/ui/aspect-ratio.d.ts +2 -2
  231. package/dist/ui/avatar.cjs +1 -1
  232. package/dist/ui/avatar.d.cts +4 -4
  233. package/dist/ui/avatar.d.ts +4 -4
  234. package/dist/ui/avatar.js +1 -1
  235. package/dist/ui/badge.cjs +1 -1
  236. package/dist/ui/badge.d.cts +2 -2
  237. package/dist/ui/badge.d.ts +2 -2
  238. package/dist/ui/badge.js +1 -1
  239. package/dist/ui/breadcrumb.d.cts +8 -8
  240. package/dist/ui/breadcrumb.d.ts +8 -8
  241. package/dist/ui/button.d.cts +1 -1
  242. package/dist/ui/button.d.ts +1 -1
  243. package/dist/ui/buttons/add-new.cjs +1 -1
  244. package/dist/ui/buttons/add-new.js +1 -1
  245. package/dist/ui/buttons/refresh.cjs +1 -1
  246. package/dist/ui/buttons/refresh.js +1 -1
  247. package/dist/ui/calendar.cjs +1 -1
  248. package/dist/ui/calendar.d.cts +4 -4
  249. package/dist/ui/calendar.d.ts +4 -4
  250. package/dist/ui/calendar.js +1 -1
  251. package/dist/ui/card.d.cts +8 -8
  252. package/dist/ui/card.d.ts +8 -8
  253. package/dist/ui/carousel.d.cts +7 -7
  254. package/dist/ui/carousel.d.ts +7 -7
  255. package/dist/ui/collapsible.d.cts +4 -4
  256. package/dist/ui/collapsible.d.ts +4 -4
  257. package/dist/ui/command.cjs +1 -1
  258. package/dist/ui/command.d.cts +2 -2
  259. package/dist/ui/command.d.ts +2 -2
  260. package/dist/ui/command.js +1 -1
  261. package/dist/ui/context-menu.d.cts +16 -16
  262. package/dist/ui/context-menu.d.ts +16 -16
  263. package/dist/ui/dialog.cjs +1 -1
  264. package/dist/ui/dialog.d.cts +1 -1
  265. package/dist/ui/dialog.d.ts +1 -1
  266. package/dist/ui/dialog.js +1 -1
  267. package/dist/ui/dropdown-menu.d.cts +16 -16
  268. package/dist/ui/dropdown-menu.d.ts +16 -16
  269. package/dist/ui/file-uploader.d.cts +2 -2
  270. package/dist/ui/file-uploader.d.ts +2 -2
  271. package/dist/ui/form.cjs +1 -1
  272. package/dist/ui/form.d.cts +7 -7
  273. package/dist/ui/form.d.ts +7 -7
  274. package/dist/ui/form.js +1 -1
  275. package/dist/ui/hover-card.d.cts +4 -4
  276. package/dist/ui/hover-card.d.ts +4 -4
  277. package/dist/ui/input-otp.d.cts +5 -5
  278. package/dist/ui/input-otp.d.ts +5 -5
  279. package/dist/ui/input.cjs +1 -1
  280. package/dist/ui/input.d.cts +1 -1
  281. package/dist/ui/input.d.ts +1 -1
  282. package/dist/ui/input.js +1 -1
  283. package/dist/ui/inputs/search-input.cjs +1 -1
  284. package/dist/ui/inputs/search-input.d.cts +3 -3
  285. package/dist/ui/inputs/search-input.d.ts +1 -1
  286. package/dist/ui/inputs/search-input.js +1 -1
  287. package/dist/ui/label.cjs +1 -1
  288. package/dist/ui/label.d.cts +2 -2
  289. package/dist/ui/label.d.ts +2 -2
  290. package/dist/ui/label.js +1 -1
  291. package/dist/ui/menubar.d.cts +17 -17
  292. package/dist/ui/menubar.d.ts +17 -17
  293. package/dist/ui/multi-select.cjs +1 -1
  294. package/dist/ui/multi-select.d.cts +2 -2
  295. package/dist/ui/multi-select.d.ts +2 -2
  296. package/dist/ui/multi-select.js +1 -1
  297. package/dist/ui/navigation-menu.d.cts +9 -9
  298. package/dist/ui/navigation-menu.d.ts +9 -9
  299. package/dist/ui/pagination.d.cts +9 -9
  300. package/dist/ui/pagination.d.ts +9 -9
  301. package/dist/ui/popover.cjs +1 -1
  302. package/dist/ui/popover.d.cts +5 -5
  303. package/dist/ui/popover.d.ts +5 -5
  304. package/dist/ui/popover.js +1 -1
  305. package/dist/ui/progress.d.cts +2 -2
  306. package/dist/ui/progress.d.ts +2 -2
  307. package/dist/ui/radio-group.d.cts +3 -3
  308. package/dist/ui/radio-group.d.ts +3 -3
  309. package/dist/ui/resizable.d.cts +4 -4
  310. package/dist/ui/resizable.d.ts +4 -4
  311. package/dist/ui/scroll-area.d.cts +3 -3
  312. package/dist/ui/scroll-area.d.ts +3 -3
  313. package/dist/ui/select.cjs +1 -1
  314. package/dist/ui/select.d.cts +11 -11
  315. package/dist/ui/select.d.ts +11 -11
  316. package/dist/ui/select.js +1 -1
  317. package/dist/ui/separator.d.cts +1 -1
  318. package/dist/ui/separator.d.ts +1 -1
  319. package/dist/ui/sheet.d.cts +9 -9
  320. package/dist/ui/sheet.d.ts +9 -9
  321. package/dist/ui/sidebar.cjs +1 -1
  322. package/dist/ui/sidebar.d.cts +28 -28
  323. package/dist/ui/sidebar.d.ts +30 -30
  324. package/dist/ui/sidebar.js +1 -1
  325. package/dist/ui/skeleton.d.cts +2 -2
  326. package/dist/ui/skeleton.d.ts +2 -2
  327. package/dist/ui/slider.d.cts +2 -2
  328. package/dist/ui/slider.d.ts +2 -2
  329. package/dist/ui/sonner.cjs +1 -1
  330. package/dist/ui/sonner.cjs.map +1 -1
  331. package/dist/ui/sonner.d.cts +3 -3
  332. package/dist/ui/sonner.d.ts +3 -3
  333. package/dist/ui/sonner.js +1 -1
  334. package/dist/ui/sonner.js.map +1 -1
  335. package/dist/ui/switch.cjs +1 -1
  336. package/dist/ui/switch.d.cts +2 -2
  337. package/dist/ui/switch.d.ts +2 -2
  338. package/dist/ui/switch.js +1 -1
  339. package/dist/ui/table.d.cts +9 -9
  340. package/dist/ui/table.d.ts +9 -9
  341. package/dist/ui/tabs.d.cts +5 -5
  342. package/dist/ui/tabs.d.ts +5 -5
  343. package/dist/ui/textarea.cjs +1 -1
  344. package/dist/ui/textarea.d.cts +2 -2
  345. package/dist/ui/textarea.d.ts +2 -2
  346. package/dist/ui/textarea.js +1 -1
  347. package/dist/ui/toggle-group.d.cts +4 -4
  348. package/dist/ui/toggle-group.d.ts +4 -4
  349. package/dist/ui/toggle.d.cts +1 -1
  350. package/dist/ui/toggle.d.ts +1 -1
  351. package/dist/ui/tooltip.cjs +1 -1
  352. package/dist/ui/tooltip.d.cts +1 -1
  353. package/dist/ui/tooltip.d.ts +1 -1
  354. package/dist/ui/tooltip.js +1 -1
  355. package/package.json +2 -2
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","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, useContext } from 'react'\n\nimport type { CredentialResponse } from '@react-oauth/google'\n\ntype Cart = {\n productUuid: string\n productName: string\n variantUuid: string\n variantName: string\n imageUrl: string\n optionValue: string\n optionTitle: 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 = useContext(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'\nimport { useDebounceCallback } from '@customafk/react-toolkit/hooks/useDebounceCallback'\nimport { cn } from '@customafk/react-toolkit/utils'\n\nimport { Minus, Plus, ShoppingBasketIcon, Trash2 } from 'lucide-react'\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'\n\nimport { useServiceLayout } from '../../hooks/use-service-layout'\n\ntype CartItemProps = {\n productUuid: string\n productName: string\n variantUuid: string\n variantName: string\n imageUrl: string\n optionValue: string\n optionTitle: string\n quantity: number\n price: number\n}\n\nexport const CartItem: React.FC<\n CartItemProps & {\n type: 'in_stock' | 'pre_order'\n }\n> = ({ type, productUuid, variantUuid, productName, variantName, imageUrl, optionValue, optionTitle, quantity, price }) => {\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?.(variantUuid, itemQuantity, type)\n }, [onUpdatingCart, variantUuid, 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?.(productUuid)\n setIsDeleting(false)\n }, [productUuid, 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 {variantName}\n {optionTitle && ` - ${optionTitle}: ${optionValue}`}\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.productUuid}-${item.variantUuid}`} {...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, useContext } 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 = useContext(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'\nimport { useIsMobile } from '@customafk/react-toolkit/hooks/useMobile'\nimport { cn } from '@customafk/react-toolkit/utils'\n\nimport { LogOutIcon, MenuIcon, ShoppingCartIcon } from 'lucide-react'\n\nimport { Button } from '@/components/ui/button'\nimport { Drawer, DrawerContent, DrawerDescription, DrawerHeader, DrawerTitle } from '@/components/ui/drawer'\nimport { Separator } from '@/components/ui/separator'\nimport { Skeleton } from '@/components/ui/skeleton'\nimport { TooltipProvider } from '@/components/ui/tooltip'\nimport { Tooltip, TooltipContent, TooltipTrigger } from '@/components/ui/tooltip'\n\nimport { useServiceLayout } from './hooks/use-service-layout'\nimport { SidebarContext, type SidebarContextProps, useServiceLayoutSidebar } from './hooks/use-service-layout-sidebar'\n\nimport { cva, type VariantProps } from 'class-variance-authority'\nimport { Slot as SlotPrimitive } from 'radix-ui'\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 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, setOpenMobile])\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, setOpenMobile, 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('bg-sidebar', 'text-sidebar-foreground', 'flex h-full w-(--sidebar-width) flex-col', className)} {...props}>\n {children}\n </aside>\n )\n }\n\n if (isMobile) {\n return (\n <Drawer direction=\"left\" open={openMobile} onOpenChange={setOpenMobile}>\n <DrawerContent\n data-sidebar=\"sidebar\"\n data-slot=\"sidebar\"\n data-mobile=\"true\"\n className=\"bg-sidebar text-sidebar-foreground w-(--sidebar-width) p-0 data-[vaul-drawer-direction=left]:w-3xs data-[vaul-drawer-direction=left]:sm:max-w-3xs [&>button]:hidden\"\n style={\n {\n '--sidebar-width': SIDEBAR_WIDTH_MOBILE,\n } as React.CSSProperties\n }\n >\n <DrawerHeader className=\"sr-only\">\n <DrawerTitle>Sidebar</DrawerTitle>\n <DrawerDescription>Displays the mobile sidebar.</DrawerDescription>\n </DrawerHeader>\n <div className=\"flex size-full flex-col\">\n <div className=\"border-border-weak flex flex-0 items-center gap-x-2 border-b p-2 pr-4\">\n <ServiceLayoutSidebarTrigger />\n <div className=\"bg-sidebar-primary text-sidebar-primary-foreground ml-2 flex aspect-square size-8 items-center justify-center rounded-lg\">\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 </DrawerContent>\n </Drawer>\n )\n }\n\n return (\n <aside\n className=\"group peer text-sidebar-foreground bg-card hidden 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 'shadow-nav fixed inset-y-0 top-14 z-10',\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-[2px]',\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('flex-1 inset-shadow-sm')}>{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=\"border-t-border mt-2 border-t\">\n <p className=\"text-muted-foreground pt-2 text-center 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('bg-sidebar-border mx-2 w-auto', 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 text-xs font-medium 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 'text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground absolute top-3.5 right-3 flex aspect-square w-5 items-center justify-center rounded-md p-0 outline-hidden transition-transform',\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 'text-left truncate',\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 'text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground peer-hover/menu-button:text-sidebar-accent-foreground absolute top-1.5 right-1 flex aspect-square w-5 items-center justify-center rounded-md p-0 outline-hidden transition-transform',\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 'text-sidebar-foreground pointer-events-none absolute right-1 flex h-5 min-w-5 items-center justify-center rounded-md px-1 text-xs font-medium tabular-nums select-none',\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 'border-sidebar-border mx-3.5 flex min-w-0 translate-x-px flex-col gap-1 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 // eslint-disable-next-line react-refresh/only-export-components\n useServiceLayoutSidebar,\n}\n","'use client'\nimport { useState } from 'react'\nimport { useMediaQuery } from '@customafk/react-toolkit/hooks/useMediaQuery'\nimport { cn } from '@customafk/react-toolkit/utils'\n\nimport { LockIcon, LogInIcon, LogOutIcon, ShoppingBag, ShoppingCartIcon, UserIcon } from 'lucide-react'\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'\nimport { GoogleLogin } from '@react-oauth/google'\n\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 children={children}\n />\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=\"bg-muted-muted size-full\">\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=\"text-text-positive truncate text-sm font-medium\">{userName}</span>\n <span className=\"text-text-positive-weak truncate text-xs font-normal\">{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 {totalItems > 0 && (\n <span className=\"bg-primary text-primary-foreground absolute -top-1 -right-1 flex h-4 min-w-4 items-center justify-center rounded-full px-1 text-xs font-bold\">\n {totalItems}\n </span>\n )}\n </Button>\n </SheetTrigger>\n <SheetContent className=\"w-[95vw] sm:max-w-md\">\n <SheetHeader className=\"border-border-weak flex-0 border-b pb-3\">\n <SheetTitle className=\"flex items-center gap-2\">\n <ShoppingBag className=\"h-5 w-5\" />\n <span>Giỏ hàng của bạn</span>\n {totalItems > 0 && <span className=\"text-text-positive-weak text-sm font-normal\">({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 'shadow-nav flex items-center',\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=\"bg-sidebar-primary text-sidebar-primary-foreground flex aspect-square size-8 items-center justify-center rounded-lg\">\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=\"bg-card size-full flex-1 p-4 pt-0\">\n <div className=\"flex flex-col items-center gap-y-1\">\n <p className=\"text-text-positive-weak text-sm\">Đă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-text-positive-weak text-sm\">Đă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=\"bg-card shadow-card flex size-full flex-col items-center justify-center gap-6 rounded-lg p-8 text-center\">\n <div className=\"bg-muted-foreground/10 flex size-20 items-center justify-center rounded-full\">\n <LockIcon className=\"text-primary size-10\" />\n </div>\n <div className=\"flex max-w-md flex-col gap-2\">\n <h2 className=\"text-2xl font-semibold\">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=\"bg-card size-full flex-1 p-4 pt-0\">\n <div className=\"flex flex-col items-center gap-y-1\">\n <p className=\"text-text-positive-weak text-sm\">Đă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-text-positive-weak text-sm\">Đă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('border-border-weak hidden w-full flex-0 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('bg-card shadow-card max-w-8xl size-full flex-1 rounded-md p-4', className)}>\n {children}\n </div>\n )\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","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, useContext } from 'react'\n\nimport type { CredentialResponse } from '@react-oauth/google'\n\ntype Cart = {\n productUuid: string\n productName: string\n variantUuid: string\n variantName: string\n imageUrl: string\n optionValue: string\n optionTitle: 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 = useContext(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'\nimport { useDebounceCallback } from '@customafk/react-toolkit/hooks/useDebounceCallback'\nimport { cn } from '@customafk/react-toolkit/utils'\n\nimport { Minus, Plus, ShoppingBasketIcon, Trash2 } from 'lucide-react'\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'\n\nimport { useServiceLayout } from '../../hooks/use-service-layout'\n\ntype CartItemProps = {\n productUuid: string\n productName: string\n variantUuid: string\n variantName: string\n imageUrl: string\n optionValue: string\n optionTitle: string\n quantity: number\n price: number\n}\n\nexport const CartItem: React.FC<\n CartItemProps & {\n type: 'in_stock' | 'pre_order'\n }\n> = ({ type, productUuid, variantUuid, productName, variantName, imageUrl, optionValue, optionTitle, quantity, price }) => {\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?.(variantUuid, itemQuantity, type)\n }, [onUpdatingCart, variantUuid, 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?.(productUuid)\n setIsDeleting(false)\n }, [productUuid, 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 {variantName}\n {optionTitle && ` - ${optionTitle}: ${optionValue}`}\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.productUuid}-${item.variantUuid}`} {...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, useContext } 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 = useContext(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'\nimport { useIsMobile } from '@customafk/react-toolkit/hooks/useMobile'\nimport { cn } from '@customafk/react-toolkit/utils'\n\nimport { LogOutIcon, MenuIcon, ShoppingCartIcon } from 'lucide-react'\n\nimport { Button } from '@/components/ui/button'\nimport { Drawer, DrawerContent, DrawerDescription, DrawerHeader, DrawerTitle } from '@/components/ui/drawer'\nimport { Separator } from '@/components/ui/separator'\nimport { Skeleton } from '@/components/ui/skeleton'\nimport { TooltipProvider } from '@/components/ui/tooltip'\nimport { Tooltip, TooltipContent, TooltipTrigger } from '@/components/ui/tooltip'\n\nimport { useServiceLayout } from './hooks/use-service-layout'\nimport { SidebarContext, type SidebarContextProps, useServiceLayoutSidebar } from './hooks/use-service-layout-sidebar'\n\nimport { cva, type VariantProps } from 'class-variance-authority'\nimport { Slot as SlotPrimitive } from 'radix-ui'\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 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, setOpenMobile])\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, setOpenMobile, 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('bg-sidebar', 'text-sidebar-foreground', 'flex h-full w-(--sidebar-width) flex-col', className)} {...props}>\n {children}\n </aside>\n )\n }\n\n if (isMobile) {\n return (\n <Drawer direction=\"left\" open={openMobile} onOpenChange={setOpenMobile}>\n <DrawerContent\n data-sidebar=\"sidebar\"\n data-slot=\"sidebar\"\n data-mobile=\"true\"\n className=\"bg-sidebar text-sidebar-foreground w-(--sidebar-width) p-0 data-[vaul-drawer-direction=left]:w-3xs data-[vaul-drawer-direction=left]:sm:max-w-3xs [&>button]:hidden\"\n style={\n {\n '--sidebar-width': SIDEBAR_WIDTH_MOBILE,\n } as React.CSSProperties\n }\n >\n <DrawerHeader className=\"sr-only\">\n <DrawerTitle>Sidebar</DrawerTitle>\n <DrawerDescription>Displays the mobile sidebar.</DrawerDescription>\n </DrawerHeader>\n <div className=\"flex size-full flex-col\">\n <div className=\"border-border-weak flex flex-0 items-center gap-x-2 border-b p-2 pr-4\">\n <ServiceLayoutSidebarTrigger />\n <div className=\"bg-sidebar-primary text-sidebar-primary-foreground ml-2 flex aspect-square size-8 items-center justify-center rounded-lg\">\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 </DrawerContent>\n </Drawer>\n )\n }\n\n return (\n <aside\n className=\"group peer text-sidebar-foreground bg-card hidden 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 'shadow-nav fixed inset-y-0 top-14 z-10',\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-[2px]',\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('flex-1 inset-shadow-sm')}>{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=\"border-t-border mt-2 border-t\">\n <p className=\"text-muted-foreground pt-2 text-center 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('bg-sidebar-border mx-2 w-auto', 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 text-xs font-medium 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 'text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground absolute top-3.5 right-3 flex aspect-square w-5 items-center justify-center rounded-md p-0 outline-hidden transition-transform',\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 'text-left truncate',\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 'text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground peer-hover/menu-button:text-sidebar-accent-foreground absolute top-1.5 right-1 flex aspect-square w-5 items-center justify-center rounded-md p-0 outline-hidden transition-transform',\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 'text-sidebar-foreground pointer-events-none absolute right-1 flex h-5 min-w-5 items-center justify-center rounded-md px-1 text-xs font-medium tabular-nums select-none',\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 'border-sidebar-border mx-3.5 flex min-w-0 translate-x-px flex-col gap-1 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 // eslint-disable-next-line react-refresh/only-export-components\n useServiceLayoutSidebar,\n}\n","'use client'\nimport { useState } from 'react'\nimport { useMediaQuery } from '@customafk/react-toolkit/hooks/useMediaQuery'\nimport { cn } from '@customafk/react-toolkit/utils'\n\nimport { LockIcon, LogInIcon, LogOutIcon, ShoppingBag, ShoppingCartIcon, UserIcon } from 'lucide-react'\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'\nimport { GoogleLogin } from '@react-oauth/google'\n\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 children={children}\n />\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=\"bg-muted-muted size-full\">\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=\"text-text-positive truncate text-sm font-medium\">{userName}</span>\n <span className=\"text-text-positive-weak truncate text-xs font-normal\">{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 {totalItems > 0 && (\n <span className=\"bg-primary text-primary-foreground absolute -top-1 -right-1 flex h-4 min-w-4 items-center justify-center rounded-full px-1 text-xs font-bold\">\n {totalItems}\n </span>\n )}\n </Button>\n </SheetTrigger>\n <SheetContent className=\"w-[95vw] sm:max-w-md\">\n <SheetHeader className=\"border-border-weak flex-0 border-b pb-3\">\n <SheetTitle className=\"flex items-center gap-2\">\n <ShoppingBag className=\"h-5 w-5\" />\n <span>Giỏ hàng của bạn</span>\n {totalItems > 0 && <span className=\"text-text-positive-weak text-sm font-normal\">({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 'shadow-nav flex items-center',\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=\"bg-sidebar-primary text-sidebar-primary-foreground flex aspect-square size-8 items-center justify-center rounded-lg\">\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=\"bg-card size-full flex-1 p-4 pt-0\">\n <div className=\"flex flex-col items-center gap-y-1\">\n <p className=\"text-text-positive-weak text-sm\">Đă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-text-positive-weak text-sm\">Đă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=\"bg-card shadow-card flex size-full flex-col items-center justify-center gap-6 rounded-lg p-8 text-center\">\n <div className=\"bg-muted-foreground/10 flex size-20 items-center justify-center rounded-full\">\n <LockIcon className=\"text-primary size-10\" />\n </div>\n <div className=\"flex max-w-md flex-col gap-2\">\n <h2 className=\"text-2xl font-semibold\">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=\"bg-card size-full flex-1 p-4 pt-0\">\n <div className=\"flex flex-col items-center gap-y-1\">\n <p className=\"text-text-positive-weak text-sm\">Đă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-text-positive-weak text-sm\">Đă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('border-border-weak hidden w-full flex-0 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('bg-card shadow-card max-w-8xl size-full flex-1 rounded-md p-4', className)}>\n {children}\n </div>\n )\n}\n"],"mappings":"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"}
@@ -1,2 +1,2 @@
1
- import{Button as e}from"./button-DhQJ9Ey8.js";import{Command as t,CommandGroup as n,CommandItem as r,CommandList as ee,CommandSeparator as i}from"./command-DhSE6UaR.js";import{PlusIcon as te,X as a}from"lucide-react";import{cn as o}from"@customafk/react-toolkit/utils";import{forwardRef as s,useCallback as c,useEffect as l,useImperativeHandle as ne,useMemo as re,useRef as u,useState as d}from"react";import{Fragment as f,jsx as p,jsxs as m}from"react/jsx-runtime";import{Command as h,useCommandState as g}from"cmdk";function _(e,t){let[n,r]=d(e);return l(()=>{let n=setTimeout(()=>r(e),t||500);return()=>{clearTimeout(n)}},[e,t]),n}function v(e,t){if(e.length===0)return{};if(!t)return{"":e};let n={};return e.forEach(e=>{let r=e[t]||``;n[r]||(n[r]=[]),n[r].push(e)}),n}function ie(e,t){let n=JSON.parse(JSON.stringify(e));for(let[e,r]of Object.entries(n))n[e]=r.filter(e=>!t.find(t=>t.value===e.value));return n}function ae(e,t){for(let[,n]of Object.entries(e))if(n.some(e=>t.find(t=>t.value===e.value)))return!0;return!1}const y=s(({className:e,...t},n)=>{let r=g(e=>e.filtered.count===0);return r?p(`div`,{ref:n,className:o(`px-2 py-4 text-center text-sm`,e),"cmdk-empty":``,role:`presentation`,...t}):null});y.displayName=`CommandEmpty`;const b=s(({value:s,onChange:g,placeholder:b,defaultOptions:x=[],options:S,delay:oe,onSearch:C,onSearchSync:w,loadingIndicator:se,emptyIndicator:T,maxSelected:E=9007199254740991,onMaxSelected:D,hidePlaceholderWhenSelected:O,disabled:k,groupBy:A,className:ce,badgeClassName:le,selectFirstItem:ue=!0,creatable:j=!1,triggerSearchOnFocus:M=!1,commandProps:N,inputProps:P,hideClearAllButton:F=!1,onAddNewItem:I},de)=>{let L=u(null),R=u(null),[z,B]=d(!1),[V,H]=d(!1),[U,W]=d(!1),[G,K]=d(s||[]),[q,J]=d(v(x,A)),[Y,X]=d(``),Z=_(Y,oe||500);ne(de,()=>({selectedValue:[...G],input:L.current,focus:()=>L?.current?.focus(),reset:()=>K([])}),[G]);let Q=c(e=>{R.current&&!R.current.contains(e.target)&&L.current&&!L.current.contains(e.target)&&(B(!1),L.current.blur())},[]),$=c(e=>{let t=G.filter(t=>t.value!==e.value);K(t),g?.(t)},[g,G]),fe=c(e=>{let t=L.current;if(t){if((e.key===`Delete`||e.key===`Backspace`)&&t.value===``&&G.length>0){let e=G[G.length-1];e.fixed||$(G[G.length-1])}e.key===`Escape`&&t.blur()}},[$,G]);l(()=>(z?(document.addEventListener(`mousedown`,Q),document.addEventListener(`touchend`,Q)):(document.removeEventListener(`mousedown`,Q),document.removeEventListener(`touchend`,Q)),()=>{document.removeEventListener(`mousedown`,Q),document.removeEventListener(`touchend`,Q)}),[z]),l(()=>{s&&K(s)},[s]),l(()=>{if(!S||C)return;let e=v(S||[],A);JSON.stringify(e)!==JSON.stringify(q)&&J(e)},[x,S,A,C,q]),l(()=>{let e=()=>{let e=w?.(Z);J(v(e||[],A))},t=async()=>{!w||!z||(M&&e(),Z&&e())};t()},[Z,A,z,M]),l(()=>{let e=async()=>{W(!0);let e=await C?.(Z);J(v(e||[],A)),W(!1)},t=async()=>{!C||!z||(M&&await e(),Z&&await e())};t()},[Z,A,z,M]);let pe=()=>{if(!j||ae(q,[{value:Y,label:Y}])||G.find(e=>e.value===Y))return;let e=p(r,{value:Y,className:`cursor-pointer`,onMouseDown:e=>{e.preventDefault(),e.stopPropagation()},onSelect:e=>{if(G.length>=E){D?.(G.length);return}X(``);let t=[...G,{value:e,label:e}];K(t),g?.(t)},children:`Create "${Y}"`});if(!C&&Y.length>0||C&&Z.length>0&&!U)return e},me=c(()=>{if(T)return C&&!j&&Object.keys(q).length===0?p(r,{value:`-`,disabled:!0,children:T}):p(y,{children:T})},[j,T,C,q]),he=re(()=>ie(q,G),[q,G]),ge=c(()=>{if(N?.filter)return N.filter;if(j)return(e,t)=>e.toLowerCase().includes(t.toLowerCase())?1:-1},[j,N?.filter]);return m(t,{ref:R,...N,onKeyDown:e=>{fe(e),N?.onKeyDown?.(e)},className:o(`h-auto overflow-visible bg-transparent`,N?.className),shouldFilter:N?.shouldFilter===void 0?!C:N.shouldFilter,filter:ge(),children:[p(`div`,{className:o(`border-border-weak relative rounded-md border text-sm transition-shadow`,`focus-within:border-primary`,`focus-within:ring-primary-weak`,`focus-within:ring-4`,`focus-within:outline-hidden`,`has-disabled:bg-muted`,`has-disabled:text-muted-foreground`,G.length!==0&&`p-1`,!k&&G.length!==0&&`cursor-text`,!F&&`pe-9`,ce),onClick:()=>{k||L?.current?.focus()},children:m(`div`,{className:`flex flex-wrap gap-1`,children:[G.map(e=>m(`div`,{className:o(`animate-fadeIn`,`bg-background`,`hover:bg-background`,`border-border-weak border border-solid`,`text-text-positive-weak relative inline-flex h-7 cursor-default items-center rounded-md ps-2 pe-7 pl-2 text-xs font-medium transition-all`,`disabled:cursor-not-allowed`,`disabled:opacity-50`,`data-fixed:pe-2`,le),"data-fixed":e.fixed,"data-disabled":k||void 0,children:[e.label,p(`button`,{type:`button`,disabled:k||e.fixed,className:o(`text-text-positive-muted absolute -inset-y-px -end-px flex size-7 items-center justify-center rounded-e-lg border border-transparent p-0 outline-0 transition-colors`,`hover:text-foreground`,`focus-visible:outline`,`focus-visible:outline-2`,`focus-visible:outline-primary-weak`,`disabled:hover:text-muted-foreground disabled:cursor-not-allowed disabled:opacity-50`),onKeyDown:t=>{t.key===`Enter`&&$(e)},onMouseDown:e=>{e.preventDefault(),e.stopPropagation()},onClick:()=>$(e),"aria-label":`Remove`,children:p(a,{size:14,strokeWidth:2,"aria-hidden":`true`})})]},e.value)),p(h.Input,{...P,ref:L,value:Y,disabled:k,onValueChange:e=>{X(e),P?.onValueChange?.(e)},onBlur:e=>{V||B(!1),P?.onBlur?.(e)},onFocus:e=>{B(!0),M&&C?.(Z),P?.onFocus?.(e)},placeholder:O&&G.length!==0?``:b,className:o(`placeholder:text-text-positive-muted flex-1 bg-transparent outline-hidden`,`disabled:placeholder:text-transparent`,{"w-full":O,"px-3 py-2":G.length===0,"ml-1":G.length!==0},P?.className)}),p(`button`,{type:`button`,onClick:()=>{K(G.filter(e=>e.fixed)),g?.(G.filter(e=>e.fixed))},className:o(`text-muted-foreground/80`,`absolute end-0 top-0 flex size-9 items-center justify-center rounded-lg border border-transparent transition-colors`,`focus-visible:outline-2`,`hover:text-text-positive`,`focus-visible:outline-primary-weak`,(F||k||G.length<1||G.filter(({fixed:e})=>e).length===G.length)&&`hidden`),"aria-label":`Clear all`,children:p(a,{size:16,strokeWidth:2,"aria-hidden":`true`})})]})}),p(`div`,{className:`relative`,children:p(`div`,{className:o(`shadow-dropdown absolute top-2 z-10 w-full overflow-hidden rounded-lg`,`data-[state=open]:animate-in`,`data-[state=closed]:animate-out`,`data-[state=closed]:fade-out-0`,`data-[state=open]:fade-in-0`,`data-[state=closed]:zoom-out-95`,`data-[state=open]:zoom-in-95`,!z&&`hidden`),"data-state":z?`open`:`closed`,children:z&&m(ee,{className:`shadow-lg outline-hidden`,onMouseLeave:()=>{H(!1)},onMouseEnter:()=>{H(!0)},onMouseUp:()=>{L?.current?.focus()},children:[I&&m(f,{children:[p(i,{}),p(n,{children:m(e,{type:`button`,variant:`ghost`,className:`w-full font-normal [&_div]:justify-start`,onClick:I,children:[p(te,{size:14,className:`opacity-60`,"aria-hidden":`true`}),`Thêm mới`]})})]}),U?se:m(f,{children:[me(),pe(),!ue&&p(r,{value:`-`,className:`hidden`}),Object.entries(he).map(([e,t])=>p(n,{heading:e,className:`h-full overflow-auto`,children:t.map(e=>p(r,{value:e.value,disabled:e.disable,onMouseDown:e=>{e.preventDefault(),e.stopPropagation()},onSelect:()=>{if(G.length>=E){D?.(G.length);return}let t=[...G,e];X(``),K(t),g?.(t)},className:o(`cursor-pointer`,e.disable&&`cursor-not-allowed opacity-50`),children:e.label},e.value))},e))]})]})})})]})});b.displayName=`MultipleSelector`;export{b as MultipleSelector,_ as useDebounce};
2
- //# sourceMappingURL=multi-select-l0HITJuv.js.map
1
+ import{Button as e}from"./button-DhQJ9Ey8.js";import{Command as t,CommandGroup as n,CommandItem as r,CommandList as ee,CommandSeparator as i}from"./command-Bdz8IRdV.js";import{PlusIcon as te,X as a}from"lucide-react";import{cn as o}from"@customafk/react-toolkit/utils";import{forwardRef as s,useCallback as c,useEffect as l,useImperativeHandle as ne,useMemo as re,useRef as u,useState as d}from"react";import{Fragment as f,jsx as p,jsxs as m}from"react/jsx-runtime";import{Command as h,useCommandState as g}from"cmdk";function _(e,t){let[n,r]=d(e);return l(()=>{let n=setTimeout(()=>r(e),t||500);return()=>{clearTimeout(n)}},[e,t]),n}function v(e,t){if(e.length===0)return{};if(!t)return{"":e};let n={};return e.forEach(e=>{let r=e[t]||``;n[r]||(n[r]=[]),n[r].push(e)}),n}function ie(e,t){let n=JSON.parse(JSON.stringify(e));for(let[e,r]of Object.entries(n))n[e]=r.filter(e=>!t.find(t=>t.value===e.value));return n}function ae(e,t){for(let[,n]of Object.entries(e))if(n.some(e=>t.find(t=>t.value===e.value)))return!0;return!1}const y=s(({className:e,...t},n)=>{let r=g(e=>e.filtered.count===0);return r?p(`div`,{ref:n,className:o(`px-2 py-4 text-center text-sm`,e),"cmdk-empty":``,role:`presentation`,...t}):null});y.displayName=`CommandEmpty`;const b=s(({value:s,onChange:g,placeholder:b,defaultOptions:x=[],options:S,delay:oe,onSearch:C,onSearchSync:w,loadingIndicator:se,emptyIndicator:T,maxSelected:E=9007199254740991,onMaxSelected:D,hidePlaceholderWhenSelected:O,disabled:k,groupBy:A,className:ce,badgeClassName:le,selectFirstItem:ue=!0,creatable:j=!1,triggerSearchOnFocus:M=!1,commandProps:N,inputProps:P,hideClearAllButton:F=!1,onAddNewItem:I},de)=>{let L=u(null),R=u(null),[z,B]=d(!1),[V,H]=d(!1),[U,W]=d(!1),[G,K]=d(s||[]),[q,J]=d(v(x,A)),[Y,X]=d(``),Z=_(Y,oe||500);ne(de,()=>({selectedValue:[...G],input:L.current,focus:()=>L?.current?.focus(),reset:()=>K([])}),[G]);let Q=c(e=>{R.current&&!R.current.contains(e.target)&&L.current&&!L.current.contains(e.target)&&(B(!1),L.current.blur())},[]),$=c(e=>{let t=G.filter(t=>t.value!==e.value);K(t),g?.(t)},[g,G]),fe=c(e=>{let t=L.current;if(t){if((e.key===`Delete`||e.key===`Backspace`)&&t.value===``&&G.length>0){let e=G[G.length-1];e.fixed||$(G[G.length-1])}e.key===`Escape`&&t.blur()}},[$,G]);l(()=>(z?(document.addEventListener(`mousedown`,Q),document.addEventListener(`touchend`,Q)):(document.removeEventListener(`mousedown`,Q),document.removeEventListener(`touchend`,Q)),()=>{document.removeEventListener(`mousedown`,Q),document.removeEventListener(`touchend`,Q)}),[z]),l(()=>{s&&K(s)},[s]),l(()=>{if(!S||C)return;let e=v(S||[],A);JSON.stringify(e)!==JSON.stringify(q)&&J(e)},[x,S,A,C,q]),l(()=>{let e=()=>{let e=w?.(Z);J(v(e||[],A))},t=async()=>{!w||!z||(M&&e(),Z&&e())};t()},[Z,A,z,M]),l(()=>{let e=async()=>{W(!0);let e=await C?.(Z);J(v(e||[],A)),W(!1)},t=async()=>{!C||!z||(M&&await e(),Z&&await e())};t()},[Z,A,z,M]);let pe=()=>{if(!j||ae(q,[{value:Y,label:Y}])||G.find(e=>e.value===Y))return;let e=p(r,{value:Y,className:`cursor-pointer`,onMouseDown:e=>{e.preventDefault(),e.stopPropagation()},onSelect:e=>{if(G.length>=E){D?.(G.length);return}X(``);let t=[...G,{value:e,label:e}];K(t),g?.(t)},children:`Create "${Y}"`});if(!C&&Y.length>0||C&&Z.length>0&&!U)return e},me=c(()=>{if(T)return C&&!j&&Object.keys(q).length===0?p(r,{value:`-`,disabled:!0,children:T}):p(y,{children:T})},[j,T,C,q]),he=re(()=>ie(q,G),[q,G]),ge=c(()=>{if(N?.filter)return N.filter;if(j)return(e,t)=>e.toLowerCase().includes(t.toLowerCase())?1:-1},[j,N?.filter]);return m(t,{ref:R,...N,onKeyDown:e=>{fe(e),N?.onKeyDown?.(e)},className:o(`h-auto overflow-visible bg-transparent`,N?.className),shouldFilter:N?.shouldFilter===void 0?!C:N.shouldFilter,filter:ge(),children:[p(`div`,{className:o(`border-border-weak relative rounded-md border text-sm transition-shadow`,`focus-within:border-primary`,`focus-within:ring-primary-weak`,`focus-within:ring-4`,`focus-within:outline-hidden`,`has-disabled:bg-muted`,`has-disabled:text-muted-foreground`,G.length!==0&&`p-1`,!k&&G.length!==0&&`cursor-text`,!F&&`pe-9`,ce),onClick:()=>{k||L?.current?.focus()},children:m(`div`,{className:`flex flex-wrap gap-1`,children:[G.map(e=>m(`div`,{className:o(`animate-fadeIn`,`bg-background`,`hover:bg-background`,`border-border-weak border border-solid`,`text-text-positive-weak relative inline-flex h-7 cursor-default items-center rounded-md ps-2 pe-7 pl-2 text-xs font-medium transition-all`,`disabled:cursor-not-allowed`,`disabled:opacity-50`,`data-fixed:pe-2`,le),"data-fixed":e.fixed,"data-disabled":k||void 0,children:[e.label,p(`button`,{type:`button`,disabled:k||e.fixed,className:o(`text-text-positive-muted absolute -inset-y-px -end-px flex size-7 items-center justify-center rounded-e-lg border border-transparent p-0 outline-0 transition-colors`,`hover:text-foreground`,`focus-visible:outline`,`focus-visible:outline-2`,`focus-visible:outline-primary-weak`,`disabled:hover:text-muted-foreground disabled:cursor-not-allowed disabled:opacity-50`),onKeyDown:t=>{t.key===`Enter`&&$(e)},onMouseDown:e=>{e.preventDefault(),e.stopPropagation()},onClick:()=>$(e),"aria-label":`Remove`,children:p(a,{size:14,strokeWidth:2,"aria-hidden":`true`})})]},e.value)),p(h.Input,{...P,ref:L,value:Y,disabled:k,onValueChange:e=>{X(e),P?.onValueChange?.(e)},onBlur:e=>{V||B(!1),P?.onBlur?.(e)},onFocus:e=>{B(!0),M&&C?.(Z),P?.onFocus?.(e)},placeholder:O&&G.length!==0?``:b,className:o(`placeholder:text-text-positive-muted flex-1 bg-transparent outline-hidden`,`disabled:placeholder:text-transparent`,{"w-full":O,"px-3 py-2":G.length===0,"ml-1":G.length!==0},P?.className)}),p(`button`,{type:`button`,onClick:()=>{K(G.filter(e=>e.fixed)),g?.(G.filter(e=>e.fixed))},className:o(`text-muted-foreground/80`,`absolute end-0 top-0 flex size-9 items-center justify-center rounded-lg border border-transparent transition-colors`,`focus-visible:outline-2`,`hover:text-text-positive`,`focus-visible:outline-primary-weak`,(F||k||G.length<1||G.filter(({fixed:e})=>e).length===G.length)&&`hidden`),"aria-label":`Clear all`,children:p(a,{size:16,strokeWidth:2,"aria-hidden":`true`})})]})}),p(`div`,{className:`relative`,children:p(`div`,{className:o(`shadow-dropdown absolute top-2 z-10 w-full overflow-hidden rounded-lg`,`data-[state=open]:animate-in`,`data-[state=closed]:animate-out`,`data-[state=closed]:fade-out-0`,`data-[state=open]:fade-in-0`,`data-[state=closed]:zoom-out-95`,`data-[state=open]:zoom-in-95`,!z&&`hidden`),"data-state":z?`open`:`closed`,children:z&&m(ee,{className:`shadow-lg outline-hidden`,onMouseLeave:()=>{H(!1)},onMouseEnter:()=>{H(!0)},onMouseUp:()=>{L?.current?.focus()},children:[I&&m(f,{children:[p(i,{}),p(n,{children:m(e,{type:`button`,variant:`ghost`,className:`w-full font-normal [&_div]:justify-start`,onClick:I,children:[p(te,{size:14,className:`opacity-60`,"aria-hidden":`true`}),`Thêm mới`]})})]}),U?se:m(f,{children:[me(),pe(),!ue&&p(r,{value:`-`,className:`hidden`}),Object.entries(he).map(([e,t])=>p(n,{heading:e,className:`h-full overflow-auto`,children:t.map(e=>p(r,{value:e.value,disabled:e.disable,onMouseDown:e=>{e.preventDefault(),e.stopPropagation()},onSelect:()=>{if(G.length>=E){D?.(G.length);return}let t=[...G,e];X(``),K(t),g?.(t)},className:o(`cursor-pointer`,e.disable&&`cursor-not-allowed opacity-50`),children:e.label},e.value))},e))]})]})})})]})});b.displayName=`MultipleSelector`;export{b as MultipleSelector,_ as useDebounce};
2
+ //# sourceMappingURL=multi-select-7gJ0p2qk.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"multi-select-l0HITJuv.js","names":["groupOption: GroupOption","value","Command","CommandPrimitive"],"sources":["../packages/components/ui/multi-select.tsx"],"sourcesContent":["'use client'\nimport { forwardRef, useCallback, useEffect, useImperativeHandle, useMemo, useRef, useState } from 'react'\nimport { cn } from '@customafk/react-toolkit/utils'\n\nimport { PlusIcon, X } from 'lucide-react'\n\nimport { Button } from '@/components/ui/button'\nimport { Command, CommandGroup, CommandItem, CommandList, CommandSeparator } from '@/components/ui/command'\n\nimport { Command as CommandPrimitive, useCommandState } from 'cmdk'\n\nexport interface Option {\n value: string\n label: string\n disable?: boolean\n /** fixed option that can&lsquo;t be removed. */\n fixed?: boolean\n /** Group the options by providing key. */\n [key: string]: string | boolean | undefined\n}\ninterface GroupOption {\n [key: string]: Option[]\n}\n\ninterface MultipleSelectorProps {\n value?: Option[]\n defaultOptions?: Option[]\n /** manually controlled options */\n options?: Option[]\n placeholder?: string\n /** Loading component. */\n loadingIndicator?: React.ReactNode\n /** Empty component. */\n emptyIndicator?: React.ReactNode\n /** Debounce time for async search. Only work with `onSearch`. */\n delay?: number\n /**\n * Only work with `onSearch` prop. Trigger search when `onFocus`.\n * For example, when user click on the input, it will trigger the search to get initial options.\n **/\n triggerSearchOnFocus?: boolean\n /** async search */\n onSearch?: (value: string) => Promise<Option[]>\n /**\n * sync search. This search will not showing loadingIndicator.\n * The rest props are the same as async search.\n * i.e.: creatable, groupBy, delay.\n **/\n onSearchSync?: (value: string) => Option[]\n onChange?: (options: Option[]) => void\n /** Limit the maximum number of selected options. */\n maxSelected?: number\n /** When the number of selected options exceeds the limit, the onMaxSelected will be called. */\n onMaxSelected?: (maxLimit: number) => void\n /** Hide the placeholder when there are options selected. */\n hidePlaceholderWhenSelected?: boolean\n disabled?: boolean\n /** Group the options base on provided key. */\n groupBy?: string\n className?: string\n badgeClassName?: string\n /**\n * First item selected is a default behavior by cmdk. That is why the default is true.\n * This is a workaround solution by add a dummy item.\n *\n * @reference: https://github.com/pacocoursey/cmdk/issues/171\n */\n selectFirstItem?: boolean\n /** Allow user to create option when there is no option matched. */\n creatable?: boolean\n /** Props of `Command` */\n commandProps?: React.ComponentPropsWithoutRef<typeof Command>\n /** Props of `CommandInput` */\n inputProps?: Omit<React.ComponentPropsWithoutRef<typeof CommandPrimitive.Input>, 'value' | 'placeholder' | 'disabled'>\n /** hide the clear all button. */\n hideClearAllButton?: boolean\n\n /** Add new item event for select with search */\n onAddNewItem?: () => void\n}\n\nexport interface MultipleSelectorRef {\n selectedValue: Option[]\n input: HTMLInputElement\n focus: () => void\n reset: () => void\n}\n\n// eslint-disable-next-line react-refresh/only-export-components\nexport function useDebounce<T>(value: T, delay?: number): T {\n const [debouncedValue, setDebouncedValue] = useState<T>(value)\n\n useEffect(() => {\n const timer = setTimeout(() => setDebouncedValue(value), delay || 500)\n\n return () => {\n clearTimeout(timer)\n }\n }, [value, delay])\n\n return debouncedValue\n}\n\nfunction transToGroupOption(options: Option[], groupBy?: string) {\n if (options.length === 0) {\n return {}\n }\n if (!groupBy) {\n return {\n '': options,\n }\n }\n\n const groupOption: GroupOption = {}\n options.forEach((option) => {\n const key = (option[groupBy] as string) || ''\n if (!groupOption[key]) {\n groupOption[key] = []\n }\n groupOption[key].push(option)\n })\n return groupOption\n}\n\nfunction removePickedOption(groupOption: GroupOption, picked: Option[]) {\n const cloneOption = JSON.parse(JSON.stringify(groupOption)) as GroupOption\n\n for (const [key, value] of Object.entries(cloneOption)) {\n cloneOption[key] = value.filter((val) => !picked.find((p) => p.value === val.value))\n }\n return cloneOption\n}\n\nfunction isOptionsExist(groupOption: GroupOption, targetOption: Option[]) {\n for (const [, value] of Object.entries(groupOption)) {\n if (value.some((option) => targetOption.find((p) => p.value === option.value))) {\n return true\n }\n }\n return false\n}\n\n/**\n * The `CommandEmpty` of shadcn/ui will cause the cmdk empty not rendering correctly.\n * So we create one and copy the `Empty` implementation from `cmdk`.\n *\n * @reference: https://github.com/hsuanyi-chou/shadcn-ui-expansions/issues/34#issuecomment-1949561607\n **/\nconst CommandEmpty = forwardRef<HTMLDivElement, React.ComponentProps<typeof CommandPrimitive.Empty>>(({ className, ...props }, forwardedRef) => {\n const render = useCommandState((state) => state.filtered.count === 0)\n\n if (!render) return null\n\n return <div ref={forwardedRef} className={cn('px-2 py-4 text-center text-sm', className)} cmdk-empty=\"\" role=\"presentation\" {...props} />\n})\n\nCommandEmpty.displayName = 'CommandEmpty'\n\nexport const MultipleSelector = forwardRef<MultipleSelectorRef, MultipleSelectorProps>(\n (\n {\n value,\n onChange,\n placeholder,\n defaultOptions: arrayDefaultOptions = [],\n options: arrayOptions,\n delay,\n onSearch,\n onSearchSync,\n loadingIndicator: LoadingIndicator,\n emptyIndicator,\n maxSelected = Number.MAX_SAFE_INTEGER,\n onMaxSelected,\n hidePlaceholderWhenSelected,\n disabled,\n groupBy,\n className,\n badgeClassName,\n selectFirstItem = true,\n creatable = false,\n triggerSearchOnFocus = false,\n commandProps,\n inputProps,\n hideClearAllButton = false,\n onAddNewItem,\n }: MultipleSelectorProps,\n ref: React.Ref<MultipleSelectorRef>,\n ) => {\n const inputRef = useRef<HTMLInputElement>(null)\n const dropdownRef = useRef<HTMLDivElement>(null) // Added this\n\n const [open, setOpen] = useState(false)\n const [onScrollbar, setOnScrollbar] = useState(false)\n const [isLoading, setIsLoading] = useState(false)\n\n const [selected, setSelected] = useState<Option[]>(value || [])\n const [options, setOptions] = useState<GroupOption>(transToGroupOption(arrayDefaultOptions, groupBy))\n const [inputValue, setInputValue] = useState('')\n\n const debouncedSearchTerm = useDebounce(inputValue, delay || 500)\n\n useImperativeHandle(\n ref,\n () => ({\n selectedValue: [...selected],\n input: inputRef.current as HTMLInputElement,\n focus: () => inputRef?.current?.focus(),\n reset: () => setSelected([]),\n }),\n [selected],\n )\n\n const handleClickOutside = useCallback((event: MouseEvent | TouchEvent) => {\n if (dropdownRef.current && !dropdownRef.current.contains(event.target as Node) && inputRef.current && !inputRef.current.contains(event.target as Node)) {\n setOpen(false)\n inputRef.current.blur()\n }\n }, [])\n\n const handleUnselect = useCallback(\n (option: Option) => {\n const newOptions = selected.filter((s) => s.value !== option.value)\n setSelected(newOptions)\n onChange?.(newOptions)\n },\n [onChange, selected],\n )\n\n const handleKeyDown = useCallback(\n (e: React.KeyboardEvent<HTMLDivElement>) => {\n const input = inputRef.current\n if (!input) return\n\n if (e.key === 'Delete' || e.key === 'Backspace') {\n if (input.value === '' && selected.length > 0) {\n const lastSelectOption = selected[selected.length - 1]\n // If last item is fixed, we should not remove it.\n if (!lastSelectOption.fixed) {\n handleUnselect(selected[selected.length - 1])\n }\n }\n }\n\n // This is not a default behavior of the <input /> field\n if (e.key === 'Escape') {\n input.blur()\n }\n },\n [handleUnselect, selected],\n )\n\n useEffect(() => {\n if (open) {\n document.addEventListener('mousedown', handleClickOutside)\n document.addEventListener('touchend', handleClickOutside)\n } else {\n document.removeEventListener('mousedown', handleClickOutside)\n document.removeEventListener('touchend', handleClickOutside)\n }\n\n return () => {\n document.removeEventListener('mousedown', handleClickOutside)\n document.removeEventListener('touchend', handleClickOutside)\n }\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [open])\n\n useEffect(() => {\n if (value) {\n setSelected(value)\n }\n }, [value])\n\n useEffect(() => {\n /** If `onSearch` is provided, do not trigger options updated. */\n if (!arrayOptions || onSearch) {\n return\n }\n const newOption = transToGroupOption(arrayOptions || [], groupBy)\n if (JSON.stringify(newOption) !== JSON.stringify(options)) {\n setOptions(newOption)\n }\n }, [arrayDefaultOptions, arrayOptions, groupBy, onSearch, options])\n\n useEffect(() => {\n /** sync search */\n\n const doSearchSync = () => {\n const res = onSearchSync?.(debouncedSearchTerm)\n setOptions(transToGroupOption(res || [], groupBy))\n }\n\n const exec = async () => {\n if (!onSearchSync || !open) return\n\n if (triggerSearchOnFocus) {\n doSearchSync()\n }\n\n if (debouncedSearchTerm) {\n doSearchSync()\n }\n }\n\n void exec()\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [debouncedSearchTerm, groupBy, open, triggerSearchOnFocus])\n\n useEffect(() => {\n /** async search */\n\n const doSearch = async () => {\n setIsLoading(true)\n const res = await onSearch?.(debouncedSearchTerm)\n setOptions(transToGroupOption(res || [], groupBy))\n setIsLoading(false)\n }\n\n const exec = async () => {\n if (!onSearch || !open) return\n\n if (triggerSearchOnFocus) {\n await doSearch()\n }\n\n if (debouncedSearchTerm) {\n await doSearch()\n }\n }\n\n void exec()\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [debouncedSearchTerm, groupBy, open, triggerSearchOnFocus])\n\n const CreatableItem = () => {\n if (!creatable) return undefined\n if (isOptionsExist(options, [{ value: inputValue, label: inputValue }]) || selected.find((s) => s.value === inputValue)) {\n return undefined\n }\n\n const Item = (\n <CommandItem\n value={inputValue}\n className=\"cursor-pointer\"\n onMouseDown={(e) => {\n e.preventDefault()\n e.stopPropagation()\n }}\n onSelect={(value: string) => {\n if (selected.length >= maxSelected) {\n onMaxSelected?.(selected.length)\n return\n }\n setInputValue('')\n const newOptions = [...selected, { value, label: value }]\n setSelected(newOptions)\n onChange?.(newOptions)\n }}\n >\n {`Create \"${inputValue}\"`}\n </CommandItem>\n )\n\n // For normal creatable\n if (!onSearch && inputValue.length > 0) {\n return Item\n }\n\n // For async search creatable. avoid showing creatable item before loading at first.\n if (onSearch && debouncedSearchTerm.length > 0 && !isLoading) {\n return Item\n }\n\n return undefined\n }\n\n const EmptyItem = useCallback(() => {\n if (!emptyIndicator) return undefined\n\n // For async search that showing emptyIndicator\n if (onSearch && !creatable && Object.keys(options).length === 0) {\n return (\n <CommandItem value=\"-\" disabled>\n {emptyIndicator}\n </CommandItem>\n )\n }\n\n return <CommandEmpty>{emptyIndicator}</CommandEmpty>\n }, [creatable, emptyIndicator, onSearch, options])\n\n const selectables = useMemo<GroupOption>(() => removePickedOption(options, selected), [options, selected])\n\n /** Avoid Creatable Selector freezing or lagging when paste a long string. */\n const commandFilter = useCallback(() => {\n if (commandProps?.filter) {\n return commandProps.filter\n }\n\n if (creatable) {\n return (value: string, search: string) => {\n return value.toLowerCase().includes(search.toLowerCase()) ? 1 : -1\n }\n }\n // Using default filter in `cmdk`. We don&lsquo;t have to provide it.\n return undefined\n }, [creatable, commandProps?.filter])\n\n return (\n <Command\n ref={dropdownRef}\n {...commandProps}\n onKeyDown={(e) => {\n handleKeyDown(e)\n commandProps?.onKeyDown?.(e)\n }}\n className={cn('h-auto overflow-visible bg-transparent', commandProps?.className)}\n shouldFilter={commandProps?.shouldFilter !== undefined ? commandProps.shouldFilter : !onSearch} // When onSearch is provided, we don&lsquo;t want to filter the options. You can still override it.\n filter={commandFilter()}\n >\n <div\n className={cn(\n 'border-border-weak relative rounded-md border text-sm transition-shadow',\n 'focus-within:border-primary',\n 'focus-within:ring-primary-weak',\n 'focus-within:ring-4',\n 'focus-within:outline-hidden',\n 'has-disabled:bg-muted',\n 'has-disabled:text-muted-foreground',\n selected.length !== 0 && 'p-1',\n !disabled && selected.length !== 0 && 'cursor-text',\n !hideClearAllButton && 'pe-9',\n className,\n )}\n onClick={() => {\n if (disabled) return\n inputRef?.current?.focus()\n }}\n >\n <div className=\"flex flex-wrap gap-1\">\n {selected.map((option) => {\n return (\n <div\n key={option.value}\n className={cn(\n 'animate-fadeIn',\n 'bg-background',\n 'hover:bg-background',\n 'border-border-weak border border-solid',\n 'text-text-positive-weak relative inline-flex h-7 cursor-default items-center rounded-md ps-2 pe-7 pl-2 text-xs font-medium transition-all',\n 'disabled:cursor-not-allowed',\n 'disabled:opacity-50',\n 'data-fixed:pe-2',\n badgeClassName,\n )}\n data-fixed={option.fixed}\n data-disabled={disabled || undefined}\n >\n {option.label}\n <button\n type=\"button\"\n disabled={disabled || option.fixed}\n className={cn(\n 'text-text-positive-muted absolute -inset-y-px -end-px flex size-7 items-center justify-center rounded-e-lg border border-transparent p-0 outline-0 transition-colors',\n 'hover:text-foreground',\n 'focus-visible:outline',\n 'focus-visible:outline-2',\n 'focus-visible:outline-primary-weak',\n 'disabled:hover:text-muted-foreground disabled:cursor-not-allowed disabled:opacity-50',\n )}\n onKeyDown={(e) => {\n if (e.key !== 'Enter') return\n handleUnselect(option)\n }}\n onMouseDown={(e) => {\n e.preventDefault()\n e.stopPropagation()\n }}\n onClick={() => handleUnselect(option)}\n aria-label=\"Remove\"\n >\n <X size={14} strokeWidth={2} aria-hidden=\"true\" />\n </button>\n </div>\n )\n })}\n {/* Avoid having the \"Search\" Icon */}\n <CommandPrimitive.Input\n {...inputProps}\n ref={inputRef}\n value={inputValue}\n disabled={disabled}\n onValueChange={(value) => {\n setInputValue(value)\n inputProps?.onValueChange?.(value)\n }}\n onBlur={(event) => {\n if (!onScrollbar) {\n setOpen(false)\n }\n inputProps?.onBlur?.(event)\n }}\n onFocus={(event) => {\n setOpen(true)\n if (triggerSearchOnFocus) {\n onSearch?.(debouncedSearchTerm)\n }\n inputProps?.onFocus?.(event)\n }}\n placeholder={hidePlaceholderWhenSelected && selected.length !== 0 ? '' : placeholder}\n className={cn(\n 'placeholder:text-text-positive-muted flex-1 bg-transparent outline-hidden',\n 'disabled:placeholder:text-transparent',\n {\n 'w-full': hidePlaceholderWhenSelected,\n 'px-3 py-2': selected.length === 0,\n 'ml-1': selected.length !== 0,\n },\n inputProps?.className,\n )}\n />\n <button\n type=\"button\"\n onClick={() => {\n setSelected(selected.filter((s) => s.fixed))\n onChange?.(selected.filter((s) => s.fixed))\n }}\n className={cn(\n 'text-muted-foreground/80',\n 'absolute end-0 top-0 flex size-9 items-center justify-center rounded-lg border border-transparent transition-colors',\n 'focus-visible:outline-2',\n 'hover:text-text-positive',\n 'focus-visible:outline-primary-weak',\n (hideClearAllButton || disabled || selected.length < 1 || selected.filter(({ fixed }) => fixed).length === selected.length) && 'hidden',\n )}\n aria-label=\"Clear all\"\n >\n <X size={16} strokeWidth={2} aria-hidden=\"true\" />\n </button>\n </div>\n </div>\n <div className=\"relative\">\n <div\n className={cn(\n 'shadow-dropdown absolute top-2 z-10 w-full overflow-hidden rounded-lg',\n 'data-[state=open]:animate-in',\n 'data-[state=closed]:animate-out',\n 'data-[state=closed]:fade-out-0',\n 'data-[state=open]:fade-in-0',\n 'data-[state=closed]:zoom-out-95',\n 'data-[state=open]:zoom-in-95',\n !open && 'hidden',\n )}\n data-state={open ? 'open' : 'closed'}\n >\n {open && (\n <CommandList\n className=\"shadow-lg outline-hidden\"\n onMouseLeave={() => {\n setOnScrollbar(false)\n }}\n onMouseEnter={() => {\n setOnScrollbar(true)\n }}\n onMouseUp={() => {\n inputRef?.current?.focus()\n }}\n >\n {onAddNewItem && (\n <>\n <CommandSeparator />\n <CommandGroup>\n <Button type=\"button\" variant=\"ghost\" className=\"w-full font-normal [&_div]:justify-start\" onClick={onAddNewItem}>\n <PlusIcon size={14} className=\"opacity-60\" aria-hidden=\"true\" />\n Thêm mới\n </Button>\n </CommandGroup>\n </>\n )}\n {isLoading ? (\n LoadingIndicator\n ) : (\n <>\n {EmptyItem()}\n {CreatableItem()}\n {!selectFirstItem && <CommandItem value=\"-\" className=\"hidden\" />}\n {Object.entries(selectables).map(([key, dropdowns]) => (\n <CommandGroup key={key} heading={key} className=\"h-full overflow-auto\">\n {dropdowns.map((option) => {\n return (\n <CommandItem\n key={option.value}\n value={option.value}\n disabled={option.disable}\n onMouseDown={(e) => {\n e.preventDefault()\n e.stopPropagation()\n }}\n onSelect={() => {\n if (selected.length >= maxSelected) {\n onMaxSelected?.(selected.length)\n return\n }\n const newOptions = [...selected, option]\n setInputValue('')\n setSelected(newOptions)\n onChange?.(newOptions)\n }}\n className={cn('cursor-pointer', option.disable && 'cursor-not-allowed opacity-50')}\n >\n {option.label}\n </CommandItem>\n )\n })}\n </CommandGroup>\n ))}\n </>\n )}\n </CommandList>\n )}\n </div>\n </div>\n </Command>\n )\n },\n)\n\nMultipleSelector.displayName = 'MultipleSelector'\n"],"mappings":"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"}
1
+ {"version":3,"file":"multi-select-7gJ0p2qk.js","names":["groupOption: GroupOption","value","Command","CommandPrimitive"],"sources":["../packages/components/ui/multi-select.tsx"],"sourcesContent":["'use client'\nimport { forwardRef, useCallback, useEffect, useImperativeHandle, useMemo, useRef, useState } from 'react'\nimport { cn } from '@customafk/react-toolkit/utils'\n\nimport { PlusIcon, X } from 'lucide-react'\n\nimport { Button } from '@/components/ui/button'\nimport { Command, CommandGroup, CommandItem, CommandList, CommandSeparator } from '@/components/ui/command'\n\nimport { Command as CommandPrimitive, useCommandState } from 'cmdk'\n\nexport interface Option {\n value: string\n label: string\n disable?: boolean\n /** fixed option that can&lsquo;t be removed. */\n fixed?: boolean\n /** Group the options by providing key. */\n [key: string]: string | boolean | undefined\n}\ninterface GroupOption {\n [key: string]: Option[]\n}\n\ninterface MultipleSelectorProps {\n value?: Option[]\n defaultOptions?: Option[]\n /** manually controlled options */\n options?: Option[]\n placeholder?: string\n /** Loading component. */\n loadingIndicator?: React.ReactNode\n /** Empty component. */\n emptyIndicator?: React.ReactNode\n /** Debounce time for async search. Only work with `onSearch`. */\n delay?: number\n /**\n * Only work with `onSearch` prop. Trigger search when `onFocus`.\n * For example, when user click on the input, it will trigger the search to get initial options.\n **/\n triggerSearchOnFocus?: boolean\n /** async search */\n onSearch?: (value: string) => Promise<Option[]>\n /**\n * sync search. This search will not showing loadingIndicator.\n * The rest props are the same as async search.\n * i.e.: creatable, groupBy, delay.\n **/\n onSearchSync?: (value: string) => Option[]\n onChange?: (options: Option[]) => void\n /** Limit the maximum number of selected options. */\n maxSelected?: number\n /** When the number of selected options exceeds the limit, the onMaxSelected will be called. */\n onMaxSelected?: (maxLimit: number) => void\n /** Hide the placeholder when there are options selected. */\n hidePlaceholderWhenSelected?: boolean\n disabled?: boolean\n /** Group the options base on provided key. */\n groupBy?: string\n className?: string\n badgeClassName?: string\n /**\n * First item selected is a default behavior by cmdk. That is why the default is true.\n * This is a workaround solution by add a dummy item.\n *\n * @reference: https://github.com/pacocoursey/cmdk/issues/171\n */\n selectFirstItem?: boolean\n /** Allow user to create option when there is no option matched. */\n creatable?: boolean\n /** Props of `Command` */\n commandProps?: React.ComponentPropsWithoutRef<typeof Command>\n /** Props of `CommandInput` */\n inputProps?: Omit<React.ComponentPropsWithoutRef<typeof CommandPrimitive.Input>, 'value' | 'placeholder' | 'disabled'>\n /** hide the clear all button. */\n hideClearAllButton?: boolean\n\n /** Add new item event for select with search */\n onAddNewItem?: () => void\n}\n\nexport interface MultipleSelectorRef {\n selectedValue: Option[]\n input: HTMLInputElement\n focus: () => void\n reset: () => void\n}\n\n// eslint-disable-next-line react-refresh/only-export-components\nexport function useDebounce<T>(value: T, delay?: number): T {\n const [debouncedValue, setDebouncedValue] = useState<T>(value)\n\n useEffect(() => {\n const timer = setTimeout(() => setDebouncedValue(value), delay || 500)\n\n return () => {\n clearTimeout(timer)\n }\n }, [value, delay])\n\n return debouncedValue\n}\n\nfunction transToGroupOption(options: Option[], groupBy?: string) {\n if (options.length === 0) {\n return {}\n }\n if (!groupBy) {\n return {\n '': options,\n }\n }\n\n const groupOption: GroupOption = {}\n options.forEach((option) => {\n const key = (option[groupBy] as string) || ''\n if (!groupOption[key]) {\n groupOption[key] = []\n }\n groupOption[key].push(option)\n })\n return groupOption\n}\n\nfunction removePickedOption(groupOption: GroupOption, picked: Option[]) {\n const cloneOption = JSON.parse(JSON.stringify(groupOption)) as GroupOption\n\n for (const [key, value] of Object.entries(cloneOption)) {\n cloneOption[key] = value.filter((val) => !picked.find((p) => p.value === val.value))\n }\n return cloneOption\n}\n\nfunction isOptionsExist(groupOption: GroupOption, targetOption: Option[]) {\n for (const [, value] of Object.entries(groupOption)) {\n if (value.some((option) => targetOption.find((p) => p.value === option.value))) {\n return true\n }\n }\n return false\n}\n\n/**\n * The `CommandEmpty` of shadcn/ui will cause the cmdk empty not rendering correctly.\n * So we create one and copy the `Empty` implementation from `cmdk`.\n *\n * @reference: https://github.com/hsuanyi-chou/shadcn-ui-expansions/issues/34#issuecomment-1949561607\n **/\nconst CommandEmpty = forwardRef<HTMLDivElement, React.ComponentProps<typeof CommandPrimitive.Empty>>(({ className, ...props }, forwardedRef) => {\n const render = useCommandState((state) => state.filtered.count === 0)\n\n if (!render) return null\n\n return <div ref={forwardedRef} className={cn('px-2 py-4 text-center text-sm', className)} cmdk-empty=\"\" role=\"presentation\" {...props} />\n})\n\nCommandEmpty.displayName = 'CommandEmpty'\n\nexport const MultipleSelector = forwardRef<MultipleSelectorRef, MultipleSelectorProps>(\n (\n {\n value,\n onChange,\n placeholder,\n defaultOptions: arrayDefaultOptions = [],\n options: arrayOptions,\n delay,\n onSearch,\n onSearchSync,\n loadingIndicator: LoadingIndicator,\n emptyIndicator,\n maxSelected = Number.MAX_SAFE_INTEGER,\n onMaxSelected,\n hidePlaceholderWhenSelected,\n disabled,\n groupBy,\n className,\n badgeClassName,\n selectFirstItem = true,\n creatable = false,\n triggerSearchOnFocus = false,\n commandProps,\n inputProps,\n hideClearAllButton = false,\n onAddNewItem,\n }: MultipleSelectorProps,\n ref: React.Ref<MultipleSelectorRef>,\n ) => {\n const inputRef = useRef<HTMLInputElement>(null)\n const dropdownRef = useRef<HTMLDivElement>(null) // Added this\n\n const [open, setOpen] = useState(false)\n const [onScrollbar, setOnScrollbar] = useState(false)\n const [isLoading, setIsLoading] = useState(false)\n\n const [selected, setSelected] = useState<Option[]>(value || [])\n const [options, setOptions] = useState<GroupOption>(transToGroupOption(arrayDefaultOptions, groupBy))\n const [inputValue, setInputValue] = useState('')\n\n const debouncedSearchTerm = useDebounce(inputValue, delay || 500)\n\n useImperativeHandle(\n ref,\n () => ({\n selectedValue: [...selected],\n input: inputRef.current as HTMLInputElement,\n focus: () => inputRef?.current?.focus(),\n reset: () => setSelected([]),\n }),\n [selected],\n )\n\n const handleClickOutside = useCallback((event: MouseEvent | TouchEvent) => {\n if (dropdownRef.current && !dropdownRef.current.contains(event.target as Node) && inputRef.current && !inputRef.current.contains(event.target as Node)) {\n setOpen(false)\n inputRef.current.blur()\n }\n }, [])\n\n const handleUnselect = useCallback(\n (option: Option) => {\n const newOptions = selected.filter((s) => s.value !== option.value)\n setSelected(newOptions)\n onChange?.(newOptions)\n },\n [onChange, selected],\n )\n\n const handleKeyDown = useCallback(\n (e: React.KeyboardEvent<HTMLDivElement>) => {\n const input = inputRef.current\n if (!input) return\n\n if (e.key === 'Delete' || e.key === 'Backspace') {\n if (input.value === '' && selected.length > 0) {\n const lastSelectOption = selected[selected.length - 1]\n // If last item is fixed, we should not remove it.\n if (!lastSelectOption.fixed) {\n handleUnselect(selected[selected.length - 1])\n }\n }\n }\n\n // This is not a default behavior of the <input /> field\n if (e.key === 'Escape') {\n input.blur()\n }\n },\n [handleUnselect, selected],\n )\n\n useEffect(() => {\n if (open) {\n document.addEventListener('mousedown', handleClickOutside)\n document.addEventListener('touchend', handleClickOutside)\n } else {\n document.removeEventListener('mousedown', handleClickOutside)\n document.removeEventListener('touchend', handleClickOutside)\n }\n\n return () => {\n document.removeEventListener('mousedown', handleClickOutside)\n document.removeEventListener('touchend', handleClickOutside)\n }\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [open])\n\n useEffect(() => {\n if (value) {\n setSelected(value)\n }\n }, [value])\n\n useEffect(() => {\n /** If `onSearch` is provided, do not trigger options updated. */\n if (!arrayOptions || onSearch) {\n return\n }\n const newOption = transToGroupOption(arrayOptions || [], groupBy)\n if (JSON.stringify(newOption) !== JSON.stringify(options)) {\n setOptions(newOption)\n }\n }, [arrayDefaultOptions, arrayOptions, groupBy, onSearch, options])\n\n useEffect(() => {\n /** sync search */\n\n const doSearchSync = () => {\n const res = onSearchSync?.(debouncedSearchTerm)\n setOptions(transToGroupOption(res || [], groupBy))\n }\n\n const exec = async () => {\n if (!onSearchSync || !open) return\n\n if (triggerSearchOnFocus) {\n doSearchSync()\n }\n\n if (debouncedSearchTerm) {\n doSearchSync()\n }\n }\n\n void exec()\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [debouncedSearchTerm, groupBy, open, triggerSearchOnFocus])\n\n useEffect(() => {\n /** async search */\n\n const doSearch = async () => {\n setIsLoading(true)\n const res = await onSearch?.(debouncedSearchTerm)\n setOptions(transToGroupOption(res || [], groupBy))\n setIsLoading(false)\n }\n\n const exec = async () => {\n if (!onSearch || !open) return\n\n if (triggerSearchOnFocus) {\n await doSearch()\n }\n\n if (debouncedSearchTerm) {\n await doSearch()\n }\n }\n\n void exec()\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [debouncedSearchTerm, groupBy, open, triggerSearchOnFocus])\n\n const CreatableItem = () => {\n if (!creatable) return undefined\n if (isOptionsExist(options, [{ value: inputValue, label: inputValue }]) || selected.find((s) => s.value === inputValue)) {\n return undefined\n }\n\n const Item = (\n <CommandItem\n value={inputValue}\n className=\"cursor-pointer\"\n onMouseDown={(e) => {\n e.preventDefault()\n e.stopPropagation()\n }}\n onSelect={(value: string) => {\n if (selected.length >= maxSelected) {\n onMaxSelected?.(selected.length)\n return\n }\n setInputValue('')\n const newOptions = [...selected, { value, label: value }]\n setSelected(newOptions)\n onChange?.(newOptions)\n }}\n >\n {`Create \"${inputValue}\"`}\n </CommandItem>\n )\n\n // For normal creatable\n if (!onSearch && inputValue.length > 0) {\n return Item\n }\n\n // For async search creatable. avoid showing creatable item before loading at first.\n if (onSearch && debouncedSearchTerm.length > 0 && !isLoading) {\n return Item\n }\n\n return undefined\n }\n\n const EmptyItem = useCallback(() => {\n if (!emptyIndicator) return undefined\n\n // For async search that showing emptyIndicator\n if (onSearch && !creatable && Object.keys(options).length === 0) {\n return (\n <CommandItem value=\"-\" disabled>\n {emptyIndicator}\n </CommandItem>\n )\n }\n\n return <CommandEmpty>{emptyIndicator}</CommandEmpty>\n }, [creatable, emptyIndicator, onSearch, options])\n\n const selectables = useMemo<GroupOption>(() => removePickedOption(options, selected), [options, selected])\n\n /** Avoid Creatable Selector freezing or lagging when paste a long string. */\n const commandFilter = useCallback(() => {\n if (commandProps?.filter) {\n return commandProps.filter\n }\n\n if (creatable) {\n return (value: string, search: string) => {\n return value.toLowerCase().includes(search.toLowerCase()) ? 1 : -1\n }\n }\n // Using default filter in `cmdk`. We don&lsquo;t have to provide it.\n return undefined\n }, [creatable, commandProps?.filter])\n\n return (\n <Command\n ref={dropdownRef}\n {...commandProps}\n onKeyDown={(e) => {\n handleKeyDown(e)\n commandProps?.onKeyDown?.(e)\n }}\n className={cn('h-auto overflow-visible bg-transparent', commandProps?.className)}\n shouldFilter={commandProps?.shouldFilter !== undefined ? commandProps.shouldFilter : !onSearch} // When onSearch is provided, we don&lsquo;t want to filter the options. You can still override it.\n filter={commandFilter()}\n >\n <div\n className={cn(\n 'border-border-weak relative rounded-md border text-sm transition-shadow',\n 'focus-within:border-primary',\n 'focus-within:ring-primary-weak',\n 'focus-within:ring-4',\n 'focus-within:outline-hidden',\n 'has-disabled:bg-muted',\n 'has-disabled:text-muted-foreground',\n selected.length !== 0 && 'p-1',\n !disabled && selected.length !== 0 && 'cursor-text',\n !hideClearAllButton && 'pe-9',\n className,\n )}\n onClick={() => {\n if (disabled) return\n inputRef?.current?.focus()\n }}\n >\n <div className=\"flex flex-wrap gap-1\">\n {selected.map((option) => {\n return (\n <div\n key={option.value}\n className={cn(\n 'animate-fadeIn',\n 'bg-background',\n 'hover:bg-background',\n 'border-border-weak border border-solid',\n 'text-text-positive-weak relative inline-flex h-7 cursor-default items-center rounded-md ps-2 pe-7 pl-2 text-xs font-medium transition-all',\n 'disabled:cursor-not-allowed',\n 'disabled:opacity-50',\n 'data-fixed:pe-2',\n badgeClassName,\n )}\n data-fixed={option.fixed}\n data-disabled={disabled || undefined}\n >\n {option.label}\n <button\n type=\"button\"\n disabled={disabled || option.fixed}\n className={cn(\n 'text-text-positive-muted absolute -inset-y-px -end-px flex size-7 items-center justify-center rounded-e-lg border border-transparent p-0 outline-0 transition-colors',\n 'hover:text-foreground',\n 'focus-visible:outline',\n 'focus-visible:outline-2',\n 'focus-visible:outline-primary-weak',\n 'disabled:hover:text-muted-foreground disabled:cursor-not-allowed disabled:opacity-50',\n )}\n onKeyDown={(e) => {\n if (e.key !== 'Enter') return\n handleUnselect(option)\n }}\n onMouseDown={(e) => {\n e.preventDefault()\n e.stopPropagation()\n }}\n onClick={() => handleUnselect(option)}\n aria-label=\"Remove\"\n >\n <X size={14} strokeWidth={2} aria-hidden=\"true\" />\n </button>\n </div>\n )\n })}\n {/* Avoid having the \"Search\" Icon */}\n <CommandPrimitive.Input\n {...inputProps}\n ref={inputRef}\n value={inputValue}\n disabled={disabled}\n onValueChange={(value) => {\n setInputValue(value)\n inputProps?.onValueChange?.(value)\n }}\n onBlur={(event) => {\n if (!onScrollbar) {\n setOpen(false)\n }\n inputProps?.onBlur?.(event)\n }}\n onFocus={(event) => {\n setOpen(true)\n if (triggerSearchOnFocus) {\n onSearch?.(debouncedSearchTerm)\n }\n inputProps?.onFocus?.(event)\n }}\n placeholder={hidePlaceholderWhenSelected && selected.length !== 0 ? '' : placeholder}\n className={cn(\n 'placeholder:text-text-positive-muted flex-1 bg-transparent outline-hidden',\n 'disabled:placeholder:text-transparent',\n {\n 'w-full': hidePlaceholderWhenSelected,\n 'px-3 py-2': selected.length === 0,\n 'ml-1': selected.length !== 0,\n },\n inputProps?.className,\n )}\n />\n <button\n type=\"button\"\n onClick={() => {\n setSelected(selected.filter((s) => s.fixed))\n onChange?.(selected.filter((s) => s.fixed))\n }}\n className={cn(\n 'text-muted-foreground/80',\n 'absolute end-0 top-0 flex size-9 items-center justify-center rounded-lg border border-transparent transition-colors',\n 'focus-visible:outline-2',\n 'hover:text-text-positive',\n 'focus-visible:outline-primary-weak',\n (hideClearAllButton || disabled || selected.length < 1 || selected.filter(({ fixed }) => fixed).length === selected.length) && 'hidden',\n )}\n aria-label=\"Clear all\"\n >\n <X size={16} strokeWidth={2} aria-hidden=\"true\" />\n </button>\n </div>\n </div>\n <div className=\"relative\">\n <div\n className={cn(\n 'shadow-dropdown absolute top-2 z-10 w-full overflow-hidden rounded-lg',\n 'data-[state=open]:animate-in',\n 'data-[state=closed]:animate-out',\n 'data-[state=closed]:fade-out-0',\n 'data-[state=open]:fade-in-0',\n 'data-[state=closed]:zoom-out-95',\n 'data-[state=open]:zoom-in-95',\n !open && 'hidden',\n )}\n data-state={open ? 'open' : 'closed'}\n >\n {open && (\n <CommandList\n className=\"shadow-lg outline-hidden\"\n onMouseLeave={() => {\n setOnScrollbar(false)\n }}\n onMouseEnter={() => {\n setOnScrollbar(true)\n }}\n onMouseUp={() => {\n inputRef?.current?.focus()\n }}\n >\n {onAddNewItem && (\n <>\n <CommandSeparator />\n <CommandGroup>\n <Button type=\"button\" variant=\"ghost\" className=\"w-full font-normal [&_div]:justify-start\" onClick={onAddNewItem}>\n <PlusIcon size={14} className=\"opacity-60\" aria-hidden=\"true\" />\n Thêm mới\n </Button>\n </CommandGroup>\n </>\n )}\n {isLoading ? (\n LoadingIndicator\n ) : (\n <>\n {EmptyItem()}\n {CreatableItem()}\n {!selectFirstItem && <CommandItem value=\"-\" className=\"hidden\" />}\n {Object.entries(selectables).map(([key, dropdowns]) => (\n <CommandGroup key={key} heading={key} className=\"h-full overflow-auto\">\n {dropdowns.map((option) => {\n return (\n <CommandItem\n key={option.value}\n value={option.value}\n disabled={option.disable}\n onMouseDown={(e) => {\n e.preventDefault()\n e.stopPropagation()\n }}\n onSelect={() => {\n if (selected.length >= maxSelected) {\n onMaxSelected?.(selected.length)\n return\n }\n const newOptions = [...selected, option]\n setInputValue('')\n setSelected(newOptions)\n onChange?.(newOptions)\n }}\n className={cn('cursor-pointer', option.disable && 'cursor-not-allowed opacity-50')}\n >\n {option.label}\n </CommandItem>\n )\n })}\n </CommandGroup>\n ))}\n </>\n )}\n </CommandList>\n )}\n </div>\n </div>\n </Command>\n )\n },\n)\n\nMultipleSelector.displayName = 'MultipleSelector'\n"],"mappings":"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"}
@@ -1,2 +1,2 @@
1
- const e=require(`./chunk-CUT6urMc.cjs`),t=require(`./button-CuV2l4ge.cjs`),n=require(`./command-DHGZlBKd.cjs`),r=e.__toESM(require(`lucide-react`)),i=e.__toESM(require(`@customafk/react-toolkit/utils`)),a=e.__toESM(require(`react`)),o=e.__toESM(require(`react/jsx-runtime`)),s=e.__toESM(require(`cmdk`));function c(e,t){let[n,r]=(0,a.useState)(e);return(0,a.useEffect)(()=>{let n=setTimeout(()=>r(e),t||500);return()=>{clearTimeout(n)}},[e,t]),n}function l(e,t){if(e.length===0)return{};if(!t)return{"":e};let n={};return e.forEach(e=>{let r=e[t]||``;n[r]||(n[r]=[]),n[r].push(e)}),n}function u(e,t){let n=JSON.parse(JSON.stringify(e));for(let[e,r]of Object.entries(n))n[e]=r.filter(e=>!t.find(t=>t.value===e.value));return n}function d(e,t){for(let[,n]of Object.entries(e))if(n.some(e=>t.find(t=>t.value===e.value)))return!0;return!1}const f=(0,a.forwardRef)(({className:e,...t},n)=>{let r=(0,s.useCommandState)(e=>e.filtered.count===0);return r?(0,o.jsx)(`div`,{ref:n,className:(0,i.cn)(`px-2 py-4 text-center text-sm`,e),"cmdk-empty":``,role:`presentation`,...t}):null});f.displayName=`CommandEmpty`;const p=(0,a.forwardRef)(({value:e,onChange:p,placeholder:m,defaultOptions:h=[],options:g,delay:_,onSearch:v,onSearchSync:y,loadingIndicator:b,emptyIndicator:x,maxSelected:S=9007199254740991,onMaxSelected:C,hidePlaceholderWhenSelected:w,disabled:T,groupBy:E,className:D,badgeClassName:O,selectFirstItem:k=!0,creatable:A=!1,triggerSearchOnFocus:j=!1,commandProps:M,inputProps:N,hideClearAllButton:P=!1,onAddNewItem:F},I)=>{let L=(0,a.useRef)(null),R=(0,a.useRef)(null),[z,B]=(0,a.useState)(!1),[V,H]=(0,a.useState)(!1),[U,W]=(0,a.useState)(!1),[G,K]=(0,a.useState)(e||[]),[q,J]=(0,a.useState)(l(h,E)),[Y,X]=(0,a.useState)(``),Z=c(Y,_||500);(0,a.useImperativeHandle)(I,()=>({selectedValue:[...G],input:L.current,focus:()=>L?.current?.focus(),reset:()=>K([])}),[G]);let Q=(0,a.useCallback)(e=>{R.current&&!R.current.contains(e.target)&&L.current&&!L.current.contains(e.target)&&(B(!1),L.current.blur())},[]),$=(0,a.useCallback)(e=>{let t=G.filter(t=>t.value!==e.value);K(t),p?.(t)},[p,G]),ee=(0,a.useCallback)(e=>{let t=L.current;if(t){if((e.key===`Delete`||e.key===`Backspace`)&&t.value===``&&G.length>0){let e=G[G.length-1];e.fixed||$(G[G.length-1])}e.key===`Escape`&&t.blur()}},[$,G]);(0,a.useEffect)(()=>(z?(document.addEventListener(`mousedown`,Q),document.addEventListener(`touchend`,Q)):(document.removeEventListener(`mousedown`,Q),document.removeEventListener(`touchend`,Q)),()=>{document.removeEventListener(`mousedown`,Q),document.removeEventListener(`touchend`,Q)}),[z]),(0,a.useEffect)(()=>{e&&K(e)},[e]),(0,a.useEffect)(()=>{if(!g||v)return;let e=l(g||[],E);JSON.stringify(e)!==JSON.stringify(q)&&J(e)},[h,g,E,v,q]),(0,a.useEffect)(()=>{let e=()=>{let e=y?.(Z);J(l(e||[],E))},t=async()=>{!y||!z||(j&&e(),Z&&e())};t()},[Z,E,z,j]),(0,a.useEffect)(()=>{let e=async()=>{W(!0);let e=await v?.(Z);J(l(e||[],E)),W(!1)},t=async()=>{!v||!z||(j&&await e(),Z&&await e())};t()},[Z,E,z,j]);let te=()=>{if(!A||d(q,[{value:Y,label:Y}])||G.find(e=>e.value===Y))return;let e=(0,o.jsx)(n.CommandItem,{value:Y,className:`cursor-pointer`,onMouseDown:e=>{e.preventDefault(),e.stopPropagation()},onSelect:e=>{if(G.length>=S){C?.(G.length);return}X(``);let t=[...G,{value:e,label:e}];K(t),p?.(t)},children:`Create "${Y}"`});if(!v&&Y.length>0||v&&Z.length>0&&!U)return e},ne=(0,a.useCallback)(()=>{if(x)return v&&!A&&Object.keys(q).length===0?(0,o.jsx)(n.CommandItem,{value:`-`,disabled:!0,children:x}):(0,o.jsx)(f,{children:x})},[A,x,v,q]),re=(0,a.useMemo)(()=>u(q,G),[q,G]),ie=(0,a.useCallback)(()=>{if(M?.filter)return M.filter;if(A)return(e,t)=>e.toLowerCase().includes(t.toLowerCase())?1:-1},[A,M?.filter]);return(0,o.jsxs)(n.Command,{ref:R,...M,onKeyDown:e=>{ee(e),M?.onKeyDown?.(e)},className:(0,i.cn)(`h-auto overflow-visible bg-transparent`,M?.className),shouldFilter:M?.shouldFilter===void 0?!v:M.shouldFilter,filter:ie(),children:[(0,o.jsx)(`div`,{className:(0,i.cn)(`border-border-weak relative rounded-md border text-sm transition-shadow`,`focus-within:border-primary`,`focus-within:ring-primary-weak`,`focus-within:ring-4`,`focus-within:outline-hidden`,`has-disabled:bg-muted`,`has-disabled:text-muted-foreground`,G.length!==0&&`p-1`,!T&&G.length!==0&&`cursor-text`,!P&&`pe-9`,D),onClick:()=>{T||L?.current?.focus()},children:(0,o.jsxs)(`div`,{className:`flex flex-wrap gap-1`,children:[G.map(e=>(0,o.jsxs)(`div`,{className:(0,i.cn)(`animate-fadeIn`,`bg-background`,`hover:bg-background`,`border-border-weak border border-solid`,`text-text-positive-weak relative inline-flex h-7 cursor-default items-center rounded-md ps-2 pe-7 pl-2 text-xs font-medium transition-all`,`disabled:cursor-not-allowed`,`disabled:opacity-50`,`data-fixed:pe-2`,O),"data-fixed":e.fixed,"data-disabled":T||void 0,children:[e.label,(0,o.jsx)(`button`,{type:`button`,disabled:T||e.fixed,className:(0,i.cn)(`text-text-positive-muted absolute -inset-y-px -end-px flex size-7 items-center justify-center rounded-e-lg border border-transparent p-0 outline-0 transition-colors`,`hover:text-foreground`,`focus-visible:outline`,`focus-visible:outline-2`,`focus-visible:outline-primary-weak`,`disabled:hover:text-muted-foreground disabled:cursor-not-allowed disabled:opacity-50`),onKeyDown:t=>{t.key===`Enter`&&$(e)},onMouseDown:e=>{e.preventDefault(),e.stopPropagation()},onClick:()=>$(e),"aria-label":`Remove`,children:(0,o.jsx)(r.X,{size:14,strokeWidth:2,"aria-hidden":`true`})})]},e.value)),(0,o.jsx)(s.Command.Input,{...N,ref:L,value:Y,disabled:T,onValueChange:e=>{X(e),N?.onValueChange?.(e)},onBlur:e=>{V||B(!1),N?.onBlur?.(e)},onFocus:e=>{B(!0),j&&v?.(Z),N?.onFocus?.(e)},placeholder:w&&G.length!==0?``:m,className:(0,i.cn)(`placeholder:text-text-positive-muted flex-1 bg-transparent outline-hidden`,`disabled:placeholder:text-transparent`,{"w-full":w,"px-3 py-2":G.length===0,"ml-1":G.length!==0},N?.className)}),(0,o.jsx)(`button`,{type:`button`,onClick:()=>{K(G.filter(e=>e.fixed)),p?.(G.filter(e=>e.fixed))},className:(0,i.cn)(`text-muted-foreground/80`,`absolute end-0 top-0 flex size-9 items-center justify-center rounded-lg border border-transparent transition-colors`,`focus-visible:outline-2`,`hover:text-text-positive`,`focus-visible:outline-primary-weak`,(P||T||G.length<1||G.filter(({fixed:e})=>e).length===G.length)&&`hidden`),"aria-label":`Clear all`,children:(0,o.jsx)(r.X,{size:16,strokeWidth:2,"aria-hidden":`true`})})]})}),(0,o.jsx)(`div`,{className:`relative`,children:(0,o.jsx)(`div`,{className:(0,i.cn)(`shadow-dropdown absolute top-2 z-10 w-full overflow-hidden rounded-lg`,`data-[state=open]:animate-in`,`data-[state=closed]:animate-out`,`data-[state=closed]:fade-out-0`,`data-[state=open]:fade-in-0`,`data-[state=closed]:zoom-out-95`,`data-[state=open]:zoom-in-95`,!z&&`hidden`),"data-state":z?`open`:`closed`,children:z&&(0,o.jsxs)(n.CommandList,{className:`shadow-lg outline-hidden`,onMouseLeave:()=>{H(!1)},onMouseEnter:()=>{H(!0)},onMouseUp:()=>{L?.current?.focus()},children:[F&&(0,o.jsxs)(o.Fragment,{children:[(0,o.jsx)(n.CommandSeparator,{}),(0,o.jsx)(n.CommandGroup,{children:(0,o.jsxs)(t.Button,{type:`button`,variant:`ghost`,className:`w-full font-normal [&_div]:justify-start`,onClick:F,children:[(0,o.jsx)(r.PlusIcon,{size:14,className:`opacity-60`,"aria-hidden":`true`}),`Thêm mới`]})})]}),U?b:(0,o.jsxs)(o.Fragment,{children:[ne(),te(),!k&&(0,o.jsx)(n.CommandItem,{value:`-`,className:`hidden`}),Object.entries(re).map(([e,t])=>(0,o.jsx)(n.CommandGroup,{heading:e,className:`h-full overflow-auto`,children:t.map(e=>(0,o.jsx)(n.CommandItem,{value:e.value,disabled:e.disable,onMouseDown:e=>{e.preventDefault(),e.stopPropagation()},onSelect:()=>{if(G.length>=S){C?.(G.length);return}let t=[...G,e];X(``),K(t),p?.(t)},className:(0,i.cn)(`cursor-pointer`,e.disable&&`cursor-not-allowed opacity-50`),children:e.label},e.value))},e))]})]})})})]})});p.displayName=`MultipleSelector`,Object.defineProperty(exports,`MultipleSelector`,{enumerable:!0,get:function(){return p}}),Object.defineProperty(exports,`useDebounce`,{enumerable:!0,get:function(){return c}});
2
- //# sourceMappingURL=multi-select-C0dFenI-.cjs.map
1
+ const e=require(`./chunk-CUT6urMc.cjs`),t=require(`./button-CuV2l4ge.cjs`),n=require(`./command-BFIF8llT.cjs`),r=e.__toESM(require(`lucide-react`)),i=e.__toESM(require(`@customafk/react-toolkit/utils`)),a=e.__toESM(require(`react`)),o=e.__toESM(require(`react/jsx-runtime`)),s=e.__toESM(require(`cmdk`));function c(e,t){let[n,r]=(0,a.useState)(e);return(0,a.useEffect)(()=>{let n=setTimeout(()=>r(e),t||500);return()=>{clearTimeout(n)}},[e,t]),n}function l(e,t){if(e.length===0)return{};if(!t)return{"":e};let n={};return e.forEach(e=>{let r=e[t]||``;n[r]||(n[r]=[]),n[r].push(e)}),n}function u(e,t){let n=JSON.parse(JSON.stringify(e));for(let[e,r]of Object.entries(n))n[e]=r.filter(e=>!t.find(t=>t.value===e.value));return n}function d(e,t){for(let[,n]of Object.entries(e))if(n.some(e=>t.find(t=>t.value===e.value)))return!0;return!1}const f=(0,a.forwardRef)(({className:e,...t},n)=>{let r=(0,s.useCommandState)(e=>e.filtered.count===0);return r?(0,o.jsx)(`div`,{ref:n,className:(0,i.cn)(`px-2 py-4 text-center text-sm`,e),"cmdk-empty":``,role:`presentation`,...t}):null});f.displayName=`CommandEmpty`;const p=(0,a.forwardRef)(({value:e,onChange:p,placeholder:m,defaultOptions:h=[],options:g,delay:_,onSearch:v,onSearchSync:y,loadingIndicator:b,emptyIndicator:x,maxSelected:S=9007199254740991,onMaxSelected:C,hidePlaceholderWhenSelected:w,disabled:T,groupBy:E,className:D,badgeClassName:O,selectFirstItem:k=!0,creatable:A=!1,triggerSearchOnFocus:j=!1,commandProps:M,inputProps:N,hideClearAllButton:P=!1,onAddNewItem:F},I)=>{let L=(0,a.useRef)(null),R=(0,a.useRef)(null),[z,B]=(0,a.useState)(!1),[V,H]=(0,a.useState)(!1),[U,W]=(0,a.useState)(!1),[G,K]=(0,a.useState)(e||[]),[q,J]=(0,a.useState)(l(h,E)),[Y,X]=(0,a.useState)(``),Z=c(Y,_||500);(0,a.useImperativeHandle)(I,()=>({selectedValue:[...G],input:L.current,focus:()=>L?.current?.focus(),reset:()=>K([])}),[G]);let Q=(0,a.useCallback)(e=>{R.current&&!R.current.contains(e.target)&&L.current&&!L.current.contains(e.target)&&(B(!1),L.current.blur())},[]),$=(0,a.useCallback)(e=>{let t=G.filter(t=>t.value!==e.value);K(t),p?.(t)},[p,G]),ee=(0,a.useCallback)(e=>{let t=L.current;if(t){if((e.key===`Delete`||e.key===`Backspace`)&&t.value===``&&G.length>0){let e=G[G.length-1];e.fixed||$(G[G.length-1])}e.key===`Escape`&&t.blur()}},[$,G]);(0,a.useEffect)(()=>(z?(document.addEventListener(`mousedown`,Q),document.addEventListener(`touchend`,Q)):(document.removeEventListener(`mousedown`,Q),document.removeEventListener(`touchend`,Q)),()=>{document.removeEventListener(`mousedown`,Q),document.removeEventListener(`touchend`,Q)}),[z]),(0,a.useEffect)(()=>{e&&K(e)},[e]),(0,a.useEffect)(()=>{if(!g||v)return;let e=l(g||[],E);JSON.stringify(e)!==JSON.stringify(q)&&J(e)},[h,g,E,v,q]),(0,a.useEffect)(()=>{let e=()=>{let e=y?.(Z);J(l(e||[],E))},t=async()=>{!y||!z||(j&&e(),Z&&e())};t()},[Z,E,z,j]),(0,a.useEffect)(()=>{let e=async()=>{W(!0);let e=await v?.(Z);J(l(e||[],E)),W(!1)},t=async()=>{!v||!z||(j&&await e(),Z&&await e())};t()},[Z,E,z,j]);let te=()=>{if(!A||d(q,[{value:Y,label:Y}])||G.find(e=>e.value===Y))return;let e=(0,o.jsx)(n.CommandItem,{value:Y,className:`cursor-pointer`,onMouseDown:e=>{e.preventDefault(),e.stopPropagation()},onSelect:e=>{if(G.length>=S){C?.(G.length);return}X(``);let t=[...G,{value:e,label:e}];K(t),p?.(t)},children:`Create "${Y}"`});if(!v&&Y.length>0||v&&Z.length>0&&!U)return e},ne=(0,a.useCallback)(()=>{if(x)return v&&!A&&Object.keys(q).length===0?(0,o.jsx)(n.CommandItem,{value:`-`,disabled:!0,children:x}):(0,o.jsx)(f,{children:x})},[A,x,v,q]),re=(0,a.useMemo)(()=>u(q,G),[q,G]),ie=(0,a.useCallback)(()=>{if(M?.filter)return M.filter;if(A)return(e,t)=>e.toLowerCase().includes(t.toLowerCase())?1:-1},[A,M?.filter]);return(0,o.jsxs)(n.Command,{ref:R,...M,onKeyDown:e=>{ee(e),M?.onKeyDown?.(e)},className:(0,i.cn)(`h-auto overflow-visible bg-transparent`,M?.className),shouldFilter:M?.shouldFilter===void 0?!v:M.shouldFilter,filter:ie(),children:[(0,o.jsx)(`div`,{className:(0,i.cn)(`border-border-weak relative rounded-md border text-sm transition-shadow`,`focus-within:border-primary`,`focus-within:ring-primary-weak`,`focus-within:ring-4`,`focus-within:outline-hidden`,`has-disabled:bg-muted`,`has-disabled:text-muted-foreground`,G.length!==0&&`p-1`,!T&&G.length!==0&&`cursor-text`,!P&&`pe-9`,D),onClick:()=>{T||L?.current?.focus()},children:(0,o.jsxs)(`div`,{className:`flex flex-wrap gap-1`,children:[G.map(e=>(0,o.jsxs)(`div`,{className:(0,i.cn)(`animate-fadeIn`,`bg-background`,`hover:bg-background`,`border-border-weak border border-solid`,`text-text-positive-weak relative inline-flex h-7 cursor-default items-center rounded-md ps-2 pe-7 pl-2 text-xs font-medium transition-all`,`disabled:cursor-not-allowed`,`disabled:opacity-50`,`data-fixed:pe-2`,O),"data-fixed":e.fixed,"data-disabled":T||void 0,children:[e.label,(0,o.jsx)(`button`,{type:`button`,disabled:T||e.fixed,className:(0,i.cn)(`text-text-positive-muted absolute -inset-y-px -end-px flex size-7 items-center justify-center rounded-e-lg border border-transparent p-0 outline-0 transition-colors`,`hover:text-foreground`,`focus-visible:outline`,`focus-visible:outline-2`,`focus-visible:outline-primary-weak`,`disabled:hover:text-muted-foreground disabled:cursor-not-allowed disabled:opacity-50`),onKeyDown:t=>{t.key===`Enter`&&$(e)},onMouseDown:e=>{e.preventDefault(),e.stopPropagation()},onClick:()=>$(e),"aria-label":`Remove`,children:(0,o.jsx)(r.X,{size:14,strokeWidth:2,"aria-hidden":`true`})})]},e.value)),(0,o.jsx)(s.Command.Input,{...N,ref:L,value:Y,disabled:T,onValueChange:e=>{X(e),N?.onValueChange?.(e)},onBlur:e=>{V||B(!1),N?.onBlur?.(e)},onFocus:e=>{B(!0),j&&v?.(Z),N?.onFocus?.(e)},placeholder:w&&G.length!==0?``:m,className:(0,i.cn)(`placeholder:text-text-positive-muted flex-1 bg-transparent outline-hidden`,`disabled:placeholder:text-transparent`,{"w-full":w,"px-3 py-2":G.length===0,"ml-1":G.length!==0},N?.className)}),(0,o.jsx)(`button`,{type:`button`,onClick:()=>{K(G.filter(e=>e.fixed)),p?.(G.filter(e=>e.fixed))},className:(0,i.cn)(`text-muted-foreground/80`,`absolute end-0 top-0 flex size-9 items-center justify-center rounded-lg border border-transparent transition-colors`,`focus-visible:outline-2`,`hover:text-text-positive`,`focus-visible:outline-primary-weak`,(P||T||G.length<1||G.filter(({fixed:e})=>e).length===G.length)&&`hidden`),"aria-label":`Clear all`,children:(0,o.jsx)(r.X,{size:16,strokeWidth:2,"aria-hidden":`true`})})]})}),(0,o.jsx)(`div`,{className:`relative`,children:(0,o.jsx)(`div`,{className:(0,i.cn)(`shadow-dropdown absolute top-2 z-10 w-full overflow-hidden rounded-lg`,`data-[state=open]:animate-in`,`data-[state=closed]:animate-out`,`data-[state=closed]:fade-out-0`,`data-[state=open]:fade-in-0`,`data-[state=closed]:zoom-out-95`,`data-[state=open]:zoom-in-95`,!z&&`hidden`),"data-state":z?`open`:`closed`,children:z&&(0,o.jsxs)(n.CommandList,{className:`shadow-lg outline-hidden`,onMouseLeave:()=>{H(!1)},onMouseEnter:()=>{H(!0)},onMouseUp:()=>{L?.current?.focus()},children:[F&&(0,o.jsxs)(o.Fragment,{children:[(0,o.jsx)(n.CommandSeparator,{}),(0,o.jsx)(n.CommandGroup,{children:(0,o.jsxs)(t.Button,{type:`button`,variant:`ghost`,className:`w-full font-normal [&_div]:justify-start`,onClick:F,children:[(0,o.jsx)(r.PlusIcon,{size:14,className:`opacity-60`,"aria-hidden":`true`}),`Thêm mới`]})})]}),U?b:(0,o.jsxs)(o.Fragment,{children:[ne(),te(),!k&&(0,o.jsx)(n.CommandItem,{value:`-`,className:`hidden`}),Object.entries(re).map(([e,t])=>(0,o.jsx)(n.CommandGroup,{heading:e,className:`h-full overflow-auto`,children:t.map(e=>(0,o.jsx)(n.CommandItem,{value:e.value,disabled:e.disable,onMouseDown:e=>{e.preventDefault(),e.stopPropagation()},onSelect:()=>{if(G.length>=S){C?.(G.length);return}let t=[...G,e];X(``),K(t),p?.(t)},className:(0,i.cn)(`cursor-pointer`,e.disable&&`cursor-not-allowed opacity-50`),children:e.label},e.value))},e))]})]})})})]})});p.displayName=`MultipleSelector`,Object.defineProperty(exports,`MultipleSelector`,{enumerable:!0,get:function(){return p}}),Object.defineProperty(exports,`useDebounce`,{enumerable:!0,get:function(){return c}});
2
+ //# sourceMappingURL=multi-select-zRSNabHi.cjs.map