@medusajs/dashboard 3.0.0-snapshot-20251216145629 → 3.0.0-snapshot-20260102134455

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 (290) hide show
  1. package/dist/add-locales-2OZXZCT5.mjs +81 -0
  2. package/dist/{adjust-inventory-26YXAXQL.mjs → adjust-inventory-47KGDWME.mjs} +2 -2
  3. package/dist/{api-key-management-create-EMP32G2D.mjs → api-key-management-create-EFPPZPZ7.mjs} +4 -4
  4. package/dist/{api-key-management-detail-6RCDH73M.mjs → api-key-management-detail-4IFED7SF.mjs} +15 -15
  5. package/dist/{api-key-management-edit-IBM3ZXHK.mjs → api-key-management-edit-QGVBNLUL.mjs} +4 -4
  6. package/dist/{api-key-management-list-KC5GOWAU.mjs → api-key-management-list-ROIJIIPQ.mjs} +4 -4
  7. package/dist/{api-key-management-sales-channels-LUB5G6RC.mjs → api-key-management-sales-channels-7P5L45AT.mjs} +6 -6
  8. package/dist/app.css +33 -3
  9. package/dist/app.js +18467 -16698
  10. package/dist/app.mjs +13 -13
  11. package/dist/{campaign-create-FJOECKGT.mjs → campaign-create-EZGZ4SBQ.mjs} +3 -3
  12. package/dist/{campaign-detail-5Q4BYCPX.mjs → campaign-detail-NFKLFPY4.mjs} +12 -12
  13. package/dist/{categories-metadata-J7M3XWI7.mjs → categories-metadata-G7BACZ26.mjs} +12 -12
  14. package/dist/{category-create-KHJZSC7G.mjs → category-create-7MV25HYX.mjs} +6 -6
  15. package/dist/{category-detail-S5IPXMHX.mjs → category-detail-XJQQ7UU4.mjs} +14 -14
  16. package/dist/{category-edit-CTA2EPDG.mjs → category-edit-LV7EARAB.mjs} +3 -3
  17. package/dist/{category-list-QBYJ4T3R.mjs → category-list-NIVLV5S3.mjs} +6 -6
  18. package/dist/{category-organize-SXP33XET.mjs → category-organize-SCRGWKDG.mjs} +3 -3
  19. package/dist/{category-products-KPW6BA5J.mjs → category-products-FXFSMCR6.mjs} +13 -13
  20. package/dist/{chunk-OXPE5TAY.mjs → chunk-24C2YVFU.mjs} +1 -1
  21. package/dist/{chunk-4JQR6QNW.mjs → chunk-333HK6Y6.mjs} +2 -2
  22. package/dist/{chunk-DODQ3KJT.mjs → chunk-3RRA56WC.mjs} +8 -3
  23. package/dist/{chunk-5LGRZSEH.mjs → chunk-3YPZ3R4U.mjs} +1 -1
  24. package/dist/chunk-5BUFAGKR.mjs +13 -0
  25. package/dist/{chunk-2DULKOPN.mjs → chunk-5ISRTMYH.mjs} +1 -1
  26. package/dist/{chunk-PHLCT2HA.mjs → chunk-5ZPKY5XP.mjs} +1 -1
  27. package/dist/{chunk-U6G4M5LP.mjs → chunk-6EXKKRZY.mjs} +1 -1
  28. package/dist/{chunk-GRZSG4EP.mjs → chunk-74WPAWMW.mjs} +127 -226
  29. package/dist/{chunk-UULAOAFM.mjs → chunk-772OR7OQ.mjs} +358 -242
  30. package/dist/chunk-7AXHHXCX.mjs +4 -0
  31. package/dist/{chunk-BMS2QLJY.mjs → chunk-BMJ5RYIO.mjs} +1 -1
  32. package/dist/{chunk-Y2YVTIJI.mjs → chunk-CW5PQUJ6.mjs} +1 -1
  33. package/dist/{chunk-IAV7IKJ6.mjs → chunk-DN5BRKVN.mjs} +1 -1
  34. package/dist/{chunk-RS7DWLEP.mjs → chunk-DQUXK4WW.mjs} +1 -1
  35. package/dist/{chunk-2XTBDCGE.mjs → chunk-DWH2OQFK.mjs} +1 -1
  36. package/dist/{chunk-CCQD65EY.mjs → chunk-FKNW5MLZ.mjs} +360 -288
  37. package/dist/{chunk-HTCYX4VD.mjs → chunk-G35UUG2P.mjs} +1 -1
  38. package/dist/{chunk-QKALAT7P.mjs → chunk-GGONMC2N.mjs} +1 -1
  39. package/dist/{chunk-FYWHE3W5.mjs → chunk-GJN5SXGZ.mjs} +1 -1
  40. package/dist/{chunk-CVHJAKLQ.mjs → chunk-HDO2UCKF.mjs} +1 -1
  41. package/dist/{chunk-3C6WQ7NH.mjs → chunk-IWNNQ6HA.mjs} +37 -3
  42. package/dist/{chunk-OL24RDYM.mjs → chunk-KHJL6MUQ.mjs} +8 -8
  43. package/dist/{chunk-6P4Q4AAP.mjs → chunk-KPI4WFJU.mjs} +3 -3
  44. package/dist/{chunk-AHZLMCZF.mjs → chunk-KSDXSKJ7.mjs} +1 -1
  45. package/dist/{chunk-BZKI5J2M.mjs → chunk-L7A2JIQY.mjs} +3 -3
  46. package/dist/{chunk-XY7A7GZJ.mjs → chunk-LEAMWI5H.mjs} +1 -1
  47. package/dist/{chunk-A4XYK3MY.mjs → chunk-LG5B2BVB.mjs} +2 -2
  48. package/dist/{chunk-23GTCEOV.mjs → chunk-ML7BA2NY.mjs} +1 -1
  49. package/dist/{chunk-OFN7DIZA.mjs → chunk-OYHFHBNL.mjs} +94 -7
  50. package/dist/{chunk-OSHH5GAS.mjs → chunk-Q34TRFIW.mjs} +1 -13
  51. package/dist/{chunk-UJ2TMPV4.mjs → chunk-QBIQEVTU.mjs} +0 -12
  52. package/dist/{chunk-YYOPBKME.mjs → chunk-QHRAWZGA.mjs} +3 -3
  53. package/dist/{chunk-A2WBKOXJ.mjs → chunk-SO2DE6MM.mjs} +2 -2
  54. package/dist/{chunk-EHU67PIM.mjs → chunk-TOCMU7UV.mjs} +7 -3
  55. package/dist/{chunk-YIOBBZUB.mjs → chunk-U7KANQ5A.mjs} +2 -2
  56. package/dist/{chunk-Z6BFNHEO.mjs → chunk-UCYUWNJL.mjs} +109 -62
  57. package/dist/{chunk-WYATCUOM.mjs → chunk-V2RGYIAG.mjs} +10 -10
  58. package/dist/{chunk-UWY5ZV66.mjs → chunk-VQ73B7ZZ.mjs} +1 -13
  59. package/dist/chunk-XD72PXRS.mjs +329 -0
  60. package/dist/{store-add-locales-VJ4RJ7UI.mjs → chunk-YFXI6CWD.mjs} +2 -67
  61. package/dist/{chunk-AWRCV3ME.mjs → chunk-YM4XGCHB.mjs} +1 -1
  62. package/dist/{chunk-ZMG5B4FG.mjs → chunk-ZGV5NOTE.mjs} +1 -1
  63. package/dist/{collection-add-products-FU2BS3D3.mjs → collection-add-products-YDMBU2JY.mjs} +13 -13
  64. package/dist/{collection-create-GWKWVT7B.mjs → collection-create-RXHEXU6T.mjs} +3 -3
  65. package/dist/{collection-detail-VJE7XHLV.mjs → collection-detail-75XPGVOX.mjs} +13 -13
  66. package/dist/{collection-edit-EZIO2BR5.mjs → collection-edit-CGF47VIN.mjs} +3 -3
  67. package/dist/{collection-list-IGA6SCNF.mjs → collection-list-75N5NYXQ.mjs} +18 -17
  68. package/dist/{collection-metadata-QK7MI3D2.mjs → collection-metadata-TGGC6HM2.mjs} +12 -12
  69. package/dist/{customer-detail-MOV2T3LF.mjs → customer-detail-KDLZ23B3.mjs} +17 -16
  70. package/dist/{customer-group-detail-6T7OXGQD.mjs → customer-group-detail-ZIAVMOYS.mjs} +12 -12
  71. package/dist/{customer-group-list-AJEAF5D2.mjs → customer-group-list-BDDC5VVZ.mjs} +15 -15
  72. package/dist/{customers-add-customer-group-QVTVSQYM.mjs → customers-add-customer-group-B3ASQGP6.mjs} +12 -12
  73. package/dist/{edit-inventory-item-H7DAZWIT.mjs → edit-inventory-item-OKAXQRJ5.mjs} +2 -2
  74. package/dist/{edit-inventory-item-attributes-7HTXXPGZ.mjs → edit-inventory-item-attributes-FVWDV6O5.mjs} +2 -2
  75. package/dist/{edit-reservation-OVTRZHJR.mjs → edit-reservation-3PU6YSNJ.mjs} +3 -3
  76. package/dist/{edit-rules-SMVRTCUP.mjs → edit-rules-LOJ3P4YB.mjs} +14 -14
  77. package/dist/en.json +73 -5
  78. package/dist/{inventory-create-ANYUM4P5.mjs → inventory-create-C2WWS7QR.mjs} +13 -13
  79. package/dist/{inventory-detail-ZPSEMYI2.mjs → inventory-detail-AKOZ4T3Z.mjs} +12 -12
  80. package/dist/{inventory-list-RXJPSVZE.mjs → inventory-list-DRFJIU2O.mjs} +2 -2
  81. package/dist/{inventory-metadata-FNEJ3RAT.mjs → inventory-metadata-N6CJZEWD.mjs} +12 -12
  82. package/dist/{inventory-stock-FD4ZM4BB.mjs → inventory-stock-IJ2N7LKC.mjs} +14 -14
  83. package/dist/{location-detail-N3GUZSY7.mjs → location-detail-PL4Y43RS.mjs} +16 -16
  84. package/dist/{location-fulfillment-providers-7ZUJAGNY.mjs → location-fulfillment-providers-2PLOVSXC.mjs} +17 -16
  85. package/dist/{location-sales-channels-P3QJTFDT.mjs → location-sales-channels-SZARSLWT.mjs} +5 -5
  86. package/dist/{location-service-zone-shipping-option-create-ZJ4GIBTJ.mjs → location-service-zone-shipping-option-create-CUNSAP24.mjs} +15 -15
  87. package/dist/{location-service-zone-shipping-option-edit-4CGPQ3VT.mjs → location-service-zone-shipping-option-edit-OJ462SVM.mjs} +1 -1
  88. package/dist/{location-service-zone-shipping-option-pricing-CR4BVYG3.mjs → location-service-zone-shipping-option-pricing-5HN2Z5RB.mjs} +2 -2
  89. package/dist/login-VBUNWZHI.mjs +301 -0
  90. package/dist/{manage-locations-7HH6R4UP.mjs → manage-locations-JPCSBA5D.mjs} +2 -2
  91. package/dist/{order-allocate-items-HZGGYJ42.mjs → order-allocate-items-6QHQASQJ.mjs} +4 -4
  92. package/dist/{order-create-claim-SCDJGM46.mjs → order-create-claim-ECVSH5GC.mjs} +22 -22
  93. package/dist/{order-create-edit-JIE3HDHP.mjs → order-create-edit-EVIOJIXE.mjs} +14 -13
  94. package/dist/{order-create-exchange-LQU4YN7F.mjs → order-create-exchange-ZVYYOGI3.mjs} +22 -22
  95. package/dist/{order-create-fulfillment-OWUVTZXW.mjs → order-create-fulfillment-FDIFEV6X.mjs} +16 -16
  96. package/dist/{order-create-refund-Q6HQY42R.mjs → order-create-refund-TNPGKR5P.mjs} +17 -17
  97. package/dist/{order-create-return-52GHGW5Z.mjs → order-create-return-NR6XQLDD.mjs} +9 -9
  98. package/dist/{order-create-shipment-WAGGEPRW.mjs → order-create-shipment-UP3WGRWB.mjs} +12 -12
  99. package/dist/{order-detail-PVPGEWGY.mjs → order-detail-F6D5DV74.mjs} +28 -28
  100. package/dist/{order-edit-billing-address-UM76J4KX.mjs → order-edit-billing-address-D4AJ6Z5Q.mjs} +44 -12
  101. package/dist/{order-edit-email-CL3KNOCM.mjs → order-edit-email-MXYA5BII.mjs} +12 -12
  102. package/dist/{order-edit-shipping-address-PIESTGVL.mjs → order-edit-shipping-address-GKW2ZPGR.mjs} +42 -12
  103. package/dist/{order-export-LE363ZLB.mjs → order-export-KQUI2TK7.mjs} +14 -14
  104. package/dist/{order-list-GRLQWN4L.mjs → order-list-7A5UKGXR.mjs} +11 -11
  105. package/dist/{order-metadata-FHBB7MTG.mjs → order-metadata-2IXELA7C.mjs} +12 -12
  106. package/dist/{order-receive-return-PRVKP6J2.mjs → order-receive-return-XXHOXGDC.mjs} +13 -13
  107. package/dist/{order-request-transfer-XSAGRUMT.mjs → order-request-transfer-2ENR64VR.mjs} +13 -13
  108. package/dist/{price-list-configuration-6S3MLNXQ.mjs → price-list-configuration-BZ3ZH5FY.mjs} +3 -3
  109. package/dist/{price-list-create-MFRUQADC.mjs → price-list-create-GFGNP6OS.mjs} +16 -16
  110. package/dist/{price-list-detail-Q5VG5VGW.mjs → price-list-detail-SNNOR5N7.mjs} +17 -17
  111. package/dist/{price-list-edit-53UW35L3.mjs → price-list-edit-YW5BXBF2.mjs} +3 -3
  112. package/dist/{price-list-list-DG5YEZ44.mjs → price-list-list-OF562HME.mjs} +4 -4
  113. package/dist/{price-list-prices-add-SDU5YZAT.mjs → price-list-prices-add-F7A2EV7L.mjs} +15 -15
  114. package/dist/{price-list-prices-edit-5USQR4D4.mjs → price-list-prices-edit-PQ26MOJE.mjs} +5 -5
  115. package/dist/{product-attributes-K4FGFORS.mjs → product-attributes-HO2B4OHP.mjs} +13 -13
  116. package/dist/{product-create-7YTRHZQQ.mjs → product-create-UEVW5JO4.mjs} +485 -561
  117. package/dist/{product-create-variant-ERKHTEJZ.mjs → product-create-variant-PLZ5Q2R6.mjs} +14 -14
  118. package/dist/{product-detail-32ZXNWOP.mjs → product-detail-Z7K5TGZY.mjs} +45 -66
  119. package/dist/{product-edit-FBAKONMR.mjs → product-edit-6I6JG4GP.mjs} +13 -13
  120. package/dist/{product-export-5AD7NELI.mjs → product-export-UMMROQ7T.mjs} +18 -17
  121. package/dist/{product-image-variants-edit-M6QF2RLE.mjs → product-image-variants-edit-VRG3AJID.mjs} +12 -12
  122. package/dist/{product-import-V3KQN4TV.mjs → product-import-EPS4KDID.mjs} +12 -12
  123. package/dist/{product-list-EUWZIFTM.mjs → product-list-37O7VB2L.mjs} +20 -20
  124. package/dist/{product-media-3VJ7KENL.mjs → product-media-IV5NUSLW.mjs} +3 -3
  125. package/dist/{product-metadata-GL2MVPDI.mjs → product-metadata-KTVRI7B3.mjs} +12 -12
  126. package/dist/product-option-create-C5BPN6P6.mjs +268 -0
  127. package/dist/product-option-detail-VVK3YBCE.mjs +331 -0
  128. package/dist/product-option-edit-7FQSKI2R.mjs +325 -0
  129. package/dist/product-option-list-YHGRI7OJ.mjs +268 -0
  130. package/dist/product-option-metadata-P2DHDGYR.mjs +73 -0
  131. package/dist/product-options-manage-B4N54NCR.mjs +358 -0
  132. package/dist/{product-organization-IUSVX22S.mjs → product-organization-L23V4JA7.mjs} +15 -15
  133. package/dist/{product-prices-4C36AG4R.mjs → product-prices-GUO3KM2U.mjs} +3 -3
  134. package/dist/{product-sales-channels-PPXUG4KT.mjs → product-sales-channels-BOZD3AC3.mjs} +5 -5
  135. package/dist/{product-shipping-profile-ETQFZ7DC.mjs → product-shipping-profile-FNFRDY47.mjs} +3 -3
  136. package/dist/{product-stock-VEGE6SUZ.mjs → product-stock-2DNYG73L.mjs} +14 -14
  137. package/dist/{product-tag-create-PQMDDKWH.mjs → product-tag-create-CW7GUCZW.mjs} +12 -12
  138. package/dist/{product-tag-detail-I3MBZX7U.mjs → product-tag-detail-IB5VIZ3K.mjs} +26 -25
  139. package/dist/{product-tag-edit-K3BBQLJR.mjs → product-tag-edit-ZSXD3DOC.mjs} +12 -12
  140. package/dist/{product-tag-list-JUWSOMB7.mjs → product-tag-list-MJO5PURP.mjs} +24 -23
  141. package/dist/{product-tag-metadata-MJH5LH7E.mjs → product-tag-metadata-DAKCXWZF.mjs} +12 -12
  142. package/dist/{product-type-create-DRFXTL5O.mjs → product-type-create-NRLNMABQ.mjs} +2 -2
  143. package/dist/{product-type-detail-RKHT5NBL.mjs → product-type-detail-J2IMAZW6.mjs} +14 -14
  144. package/dist/{product-type-edit-SRHCZDK7.mjs → product-type-edit-BHTPMMVI.mjs} +2 -2
  145. package/dist/{product-type-list-QQKAHBJ3.mjs → product-type-list-MQ2PW2IE.mjs} +3 -3
  146. package/dist/{product-type-metadata-CDJDFFGQ.mjs → product-type-metadata-BCZMUG3K.mjs} +12 -12
  147. package/dist/{product-variant-detail-XAYG5CKE.mjs → product-variant-detail-4L6DHT3K.mjs} +12 -12
  148. package/dist/{product-variant-edit-DEZEY2H2.mjs → product-variant-edit-PZU4P65H.mjs} +13 -13
  149. package/dist/{product-variant-manage-inventory-items-Y2VEOHP7.mjs → product-variant-manage-inventory-items-E3GPBPEZ.mjs} +3 -3
  150. package/dist/{product-variant-media-2WLVNGI4.mjs → product-variant-media-BSNCO642.mjs} +2 -2
  151. package/dist/{product-variant-metadata-VTZDNWUT.mjs → product-variant-metadata-FHGGZGPV.mjs} +12 -12
  152. package/dist/{promotion-add-campaign-DO67QK6M.mjs → promotion-add-campaign-NQSIFI33.mjs} +3 -3
  153. package/dist/{promotion-create-HWFNUQXG.mjs → promotion-create-GBZWJVAI.mjs} +24 -21
  154. package/dist/{promotion-detail-QC36KXB3.mjs → promotion-detail-W2OGUESO.mjs} +12 -12
  155. package/dist/{refund-reason-create-YHCDEHGQ.mjs → refund-reason-create-DLVRT5OU.mjs} +12 -12
  156. package/dist/{refund-reason-edit-CZ5QZ2SZ.mjs → refund-reason-edit-XCSU7RMH.mjs} +12 -12
  157. package/dist/{refund-reason-list-OJYYEYJE.mjs → refund-reason-list-G6UNVGPG.mjs} +18 -18
  158. package/dist/{region-create-NA7Y2LN4.mjs → region-create-2CRTF7HJ.mjs} +1 -1
  159. package/dist/{region-edit-WAU347DP.mjs → region-edit-IB3PJE7V.mjs} +1 -1
  160. package/dist/{region-metadata-H6XXUQ4S.mjs → region-metadata-WBEUKOI6.mjs} +12 -12
  161. package/dist/{reservation-create-ZCIYM6JI.mjs → reservation-create-ERFJUBME.mjs} +4 -4
  162. package/dist/{reservation-detail-LZAQL4XA.mjs → reservation-detail-Y6UZRZXY.mjs} +12 -12
  163. package/dist/{reservation-list-B47DXTA7.mjs → reservation-list-LEDZB7KL.mjs} +3 -3
  164. package/dist/{reservation-metadata-5HZSDDOK.mjs → reservation-metadata-7KH4EXMN.mjs} +12 -12
  165. package/dist/{return-reason-list-SCBGTOEI.mjs → return-reason-list-UUQAL464.mjs} +6 -6
  166. package/dist/{sales-channel-add-products-F7YV4MO5.mjs → sales-channel-add-products-GSPG7OTK.mjs} +14 -14
  167. package/dist/{sales-channel-create-MI7HHZYE.mjs → sales-channel-create-MM237EBQ.mjs} +3 -3
  168. package/dist/{sales-channel-detail-MXIPZCGA.mjs → sales-channel-detail-65ZFV2OI.mjs} +14 -14
  169. package/dist/{sales-channel-edit-VSHOIR37.mjs → sales-channel-edit-AC2J4N3M.mjs} +3 -3
  170. package/dist/{sales-channel-list-RLGL7FM3.mjs → sales-channel-list-LXLW7X3U.mjs} +14 -14
  171. package/dist/{sales-channel-metadata-M364R4RJ.mjs → sales-channel-metadata-QUAL2RH6.mjs} +13 -13
  172. package/dist/{shipping-option-type-create-C5WUWON7.mjs → shipping-option-type-create-EU75JC2Q.mjs} +12 -12
  173. package/dist/{shipping-option-type-detail-PENS2K73.mjs → shipping-option-type-detail-5GYEWATD.mjs} +13 -13
  174. package/dist/{shipping-option-type-edit-CIU5EHRP.mjs → shipping-option-type-edit-YR4JXT5M.mjs} +12 -12
  175. package/dist/{shipping-option-type-list-DIOX7VG7.mjs → shipping-option-type-list-BLC4Z5QW.mjs} +13 -13
  176. package/dist/{shipping-profile-metadata-75G2NNMA.mjs → shipping-profile-metadata-64YRWPDA.mjs} +12 -12
  177. package/dist/store-add-locales-TSMLNFZP.mjs +81 -0
  178. package/dist/{store-detail-JSNPOB2F.mjs → store-detail-3SS53LZR.mjs} +13 -13
  179. package/dist/{store-edit-5ZS562ZO.mjs → store-edit-SMEWFETF.mjs} +1 -1
  180. package/dist/{store-metadata-CYXTVJUE.mjs → store-metadata-IJGVDDA6.mjs} +12 -12
  181. package/dist/{tax-region-create-DWGL4EUT.mjs → tax-region-create-XLYIS5XX.mjs} +13 -13
  182. package/dist/{tax-region-detail-2AE2EFI3.mjs → tax-region-detail-66FKEGFB.mjs} +29 -28
  183. package/dist/{tax-region-edit-EEVEEU2Q.mjs → tax-region-edit-GBR4EDW7.mjs} +13 -13
  184. package/dist/{tax-region-province-detail-4ERSEQFF.mjs → tax-region-province-detail-OBA7Q7MQ.mjs} +29 -28
  185. package/dist/{tax-region-tax-override-create-PHCGEF7V.mjs → tax-region-tax-override-create-34WYD2QZ.mjs} +28 -27
  186. package/dist/{tax-region-tax-override-edit-SMRPSILC.mjs → tax-region-tax-override-edit-6WJVOGWA.mjs} +27 -26
  187. package/dist/translation-list-J65VNNAX.mjs +527 -0
  188. package/dist/{translations-edit-KECSHYZY.mjs → translations-edit-6P3KC4MO.mjs} +280 -304
  189. package/dist/{user-detail-BJUXLZZQ.mjs → user-detail-VMOJFHIJ.mjs} +2 -2
  190. package/dist/{user-metadata-2WPJOEJA.mjs → user-metadata-UOPVFX2L.mjs} +12 -12
  191. package/dist/{workflow-execution-detail-H2AKEZJX.mjs → workflow-execution-detail-WAQ7IG4T.mjs} +12 -12
  192. package/package.json +9 -9
  193. package/src/components/data-grid/components/data-grid-cell-container.tsx +7 -10
  194. package/src/components/data-grid/components/data-grid-multiline-cell.tsx +103 -0
  195. package/src/components/data-grid/components/data-grid-readonly-cell.tsx +1 -4
  196. package/src/components/data-grid/components/data-grid-root.tsx +68 -51
  197. package/src/components/data-grid/components/data-grid-text-cell.tsx +6 -76
  198. package/src/components/data-grid/components/index.ts +1 -0
  199. package/src/components/data-grid/data-grid.tsx +2 -0
  200. package/src/components/data-grid/hooks/use-data-grid-cell.tsx +12 -0
  201. package/src/components/data-grid/hooks/use-data-grid-form-handlers.tsx +1 -0
  202. package/src/components/data-grid/hooks/use-data-grid-keydown-event.tsx +65 -7
  203. package/src/components/data-grid/types.ts +1 -0
  204. package/src/components/inputs/combobox/combobox.tsx +185 -95
  205. package/src/components/layout/main-layout/main-layout.tsx +7 -4
  206. package/src/components/table/table-cells/order/country-cell/country-cell.tsx +5 -1
  207. package/src/dashboard-app/routes/get-route.map.tsx +68 -12
  208. package/src/hooks/api/cloud.tsx +41 -0
  209. package/src/hooks/api/index.ts +3 -1
  210. package/src/hooks/api/product-options.tsx +151 -0
  211. package/src/hooks/api/products.tsx +26 -66
  212. package/src/hooks/api/translations.tsx +103 -191
  213. package/src/hooks/table/columns/use-order-table-columns.tsx +5 -3
  214. package/src/hooks/table/columns/use-product-option-table-columns.tsx +44 -0
  215. package/src/hooks/table/filters/index.ts +1 -0
  216. package/src/hooks/table/filters/use-product-option-table-filters.tsx +33 -0
  217. package/src/hooks/table/query/use-product-option-table-query.tsx +34 -0
  218. package/src/i18n/translations/$schema.json +290 -32
  219. package/src/i18n/translations/en.json +73 -5
  220. package/src/i18n/translations/es.json +9 -1
  221. package/src/routes/customers/customer-detail/components/customer-order-section/customer-order-section.tsx +1 -1
  222. package/src/routes/login/components/cloud-auth-login.tsx +114 -0
  223. package/src/routes/login/login.tsx +9 -1
  224. package/src/routes/orders/order-create-edit/components/order-edit-create-form/order-edit-item.tsx +1 -0
  225. package/src/routes/orders/order-edit-billing-address/components/edit-order-billing-address-form/edit-order-billing-address-form.tsx +34 -0
  226. package/src/routes/orders/order-edit-shipping-address/components/edit-order-shipping-address-form/edit-order-shipping-address-form.tsx +32 -0
  227. package/src/routes/product-options/common/hooks/use-delete-product-option-action.tsx +41 -0
  228. package/src/routes/product-options/product-option-create/components/create-product-option-form/create-product-option-details.tsx +71 -0
  229. package/src/routes/product-options/product-option-create/components/create-product-option-form/create-product-option-form.tsx +97 -0
  230. package/src/routes/product-options/product-option-create/components/create-product-option-form/create-product-option-organize.tsx +92 -0
  231. package/src/routes/product-options/product-option-create/components/create-product-option-form/index.ts +1 -0
  232. package/src/routes/product-options/product-option-create/components/create-product-option-form/schema.ts +15 -0
  233. package/src/routes/product-options/product-option-create/index.ts +1 -0
  234. package/src/routes/product-options/product-option-create/product-option-create.tsx +10 -0
  235. package/src/routes/product-options/product-option-detail/breadcrumb.tsx +14 -0
  236. package/src/routes/product-options/product-option-detail/components/product-option-general-section/index.ts +1 -0
  237. package/src/routes/product-options/product-option-detail/components/product-option-general-section/product-option-general-section.tsx +84 -0
  238. package/src/routes/product-options/product-option-detail/components/product-option-product-section/index.ts +1 -0
  239. package/src/routes/product-options/product-option-detail/components/product-option-product-section/product-option-product-section.tsx +82 -0
  240. package/src/routes/product-options/product-option-detail/index.ts +3 -0
  241. package/src/routes/product-options/product-option-detail/loader.ts +18 -0
  242. package/src/routes/product-options/product-option-detail/product-option-detail.tsx +50 -0
  243. package/src/routes/product-options/product-option-edit/components/edit-product-option-form/edit-product-option-details.tsx +94 -0
  244. package/src/routes/product-options/product-option-edit/components/edit-product-option-form/edit-product-option-form.tsx +116 -0
  245. package/src/routes/product-options/product-option-edit/components/edit-product-option-form/edit-product-option-organize.tsx +77 -0
  246. package/src/routes/product-options/product-option-edit/components/edit-product-option-form/index.ts +1 -0
  247. package/src/routes/product-options/product-option-edit/components/edit-product-option-form/schema.ts +13 -0
  248. package/src/routes/product-options/product-option-edit/index.ts +1 -0
  249. package/src/routes/product-options/product-option-edit/product-option-edit.tsx +34 -0
  250. package/src/routes/product-options/product-option-list/components/product-option-list-table/index.ts +1 -0
  251. package/src/routes/product-options/product-option-list/components/product-option-list-table/product-option-list-table.tsx +140 -0
  252. package/src/routes/product-options/product-option-list/index.ts +1 -0
  253. package/src/routes/product-options/product-option-list/product-option-list.tsx +19 -0
  254. package/src/routes/product-options/product-option-metadata/index.ts +1 -0
  255. package/src/routes/product-options/product-option-metadata/product-option-metadata.tsx +27 -0
  256. package/src/routes/products/product-create/components/product-create-details-form/components/product-create-details-general-section/product-create-general-section.tsx +16 -4
  257. package/src/routes/products/product-create/components/product-create-details-form/components/product-create-details-variant-section/product-create-details-variant-section.tsx +327 -194
  258. package/src/routes/products/product-create/components/product-create-form/product-create-form.tsx +4 -7
  259. package/src/routes/products/product-create/constants.ts +2 -0
  260. package/src/routes/products/product-create/utils.ts +6 -1
  261. package/src/routes/products/product-detail/components/product-option-section/product-option-section.tsx +33 -51
  262. package/src/routes/products/product-options-manage/components/product-options-manage-form/index.ts +1 -0
  263. package/src/routes/products/product-options-manage/components/product-options-manage-form/product-options-manage-form.tsx +328 -0
  264. package/src/routes/products/product-options-manage/index.ts +1 -0
  265. package/src/routes/products/product-options-manage/product-options-manage.tsx +35 -0
  266. package/src/routes/promotions/promotion-create/components/create-promotion-form/create-promotion-form.tsx +6 -0
  267. package/src/routes/translations/add-locales/add-locales.tsx +29 -0
  268. package/src/routes/translations/add-locales/index.tsx +1 -0
  269. package/src/routes/translations/translation-list/components/active-locales-section/active-locales-section.tsx +42 -17
  270. package/src/routes/translations/translation-list/components/translation-list-section/translation-list-section.tsx +5 -1
  271. package/src/routes/translations/translation-list/components/translations-completion-section/translations-completion-section.tsx +182 -121
  272. package/src/routes/translations/translations-edit/components/translations-edit-form/translations-edit-form.tsx +330 -361
  273. package/src/routes/translations/translations-edit/translations-edit.tsx +8 -13
  274. package/dist/chunk-HGRIOEAR.mjs +0 -32
  275. package/dist/chunk-XDJ7OMBR.mjs +0 -160
  276. package/dist/login-VNOLI5YG.mjs +0 -201
  277. package/dist/product-create-option-7AOXAA4S.mjs +0 -151
  278. package/dist/product-edit-option-LWJT3CYJ.mjs +0 -156
  279. package/dist/translation-list-UF7FLXOW.mjs +0 -441
  280. package/src/routes/products/product-create-option/components/create-product-option-form/create-product-option-form.tsx +0 -122
  281. package/src/routes/products/product-create-option/components/create-product-option-form/index.ts +0 -1
  282. package/src/routes/products/product-create-option/index.ts +0 -1
  283. package/src/routes/products/product-create-option/product-create-option.tsx +0 -30
  284. package/src/routes/products/product-edit-option/components/edit-product-option-form/edit-product-option-form.tsx +0 -123
  285. package/src/routes/products/product-edit-option/components/edit-product-option-form/index.ts +0 -1
  286. package/src/routes/products/product-edit-option/index.ts +0 -1
  287. package/src/routes/products/product-edit-option/product-edit-option.tsx +0 -35
  288. package/dist/{chunk-FBYTX6K7.mjs → chunk-ORWRXJJT.mjs} +4 -4
  289. package/dist/{location-list-KVBA6J47.mjs → location-list-3WP65J3E.mjs} +1 -1
  290. package/dist/{user-list-YTZQNYSO.mjs → user-list-3EAKK4XC.mjs} +3 -3
@@ -0,0 +1,92 @@
1
+ import { useMemo } from "react"
2
+ import { UseFormReturn, useWatch } from "react-hook-form"
3
+ import { useTranslation } from "react-i18next"
4
+
5
+ import { Text } from "@medusajs/ui"
6
+ import { SortableList } from "../../../../../components/common/sortable-list"
7
+ import { CreateProductOptionSchema } from "./schema"
8
+
9
+ type CreateProductOptionOrganizeProps = {
10
+ form: UseFormReturn<CreateProductOptionSchema>
11
+ }
12
+
13
+ type ValueItem = {
14
+ id: string
15
+ value: string
16
+ rank: number
17
+ }
18
+
19
+ export const CreateProductOptionOrganize = ({
20
+ form,
21
+ }: CreateProductOptionOrganizeProps) => {
22
+ const { t } = useTranslation()
23
+
24
+ const values = useWatch({
25
+ control: form.control,
26
+ name: "values",
27
+ })
28
+
29
+ const valueRanks = useWatch({
30
+ control: form.control,
31
+ name: "value_ranks",
32
+ })
33
+
34
+ const items = useMemo<ValueItem[]>(() => {
35
+ if (!values || values.length === 0) {
36
+ return []
37
+ }
38
+
39
+ return values
40
+ .map((value, index) => ({
41
+ id: value,
42
+ value,
43
+ rank: valueRanks?.[value] ?? index,
44
+ }))
45
+ .sort((a, b) => a.rank - b.rank)
46
+ }, [values, valueRanks])
47
+
48
+ const handleChange = (newItems: ValueItem[]) => {
49
+ const newRanks: Record<string, number> = {}
50
+ newItems.forEach((item, index) => {
51
+ newRanks[item.value] = index
52
+ })
53
+
54
+ form.setValue("value_ranks", newRanks, {
55
+ shouldDirty: true,
56
+ shouldTouch: true,
57
+ })
58
+ }
59
+
60
+ if (!values || values.length === 0) {
61
+ return null
62
+ }
63
+
64
+ return (
65
+ <div className="px-16">
66
+ <div className="mx-auto flex w-full max-w-[720px] flex-col gap-y-8 ">
67
+ <div className="border-ui-border-base flex flex-col gap-y-2 rounded-xl border">
68
+ <div className="border-b-base bg-ui-bg-component border-b p-2">
69
+ <Text size="small" leading="compact" weight="plus">
70
+ {t("productOptions.create.tabs.organize")}
71
+ </Text>
72
+ </div>
73
+ <SortableList
74
+ items={items}
75
+ onChange={handleChange}
76
+ renderItem={(item) => (
77
+ <SortableList.Item
78
+ id={item.id}
79
+ className="border-ui-border-base border-b last:border-b-0"
80
+ >
81
+ <div className="flex flex-1 items-center gap-x-3 px-4 py-3">
82
+ <SortableList.DragHandle />
83
+ <Text size="small">{item.value}</Text>
84
+ </div>
85
+ </SortableList.Item>
86
+ )}
87
+ />{" "}
88
+ </div>
89
+ </div>
90
+ </div>
91
+ )
92
+ }
@@ -0,0 +1 @@
1
+ export { CreateProductOptionForm } from "./create-product-option-form"
@@ -0,0 +1,15 @@
1
+ import { z } from "zod"
2
+
3
+ export const CreateProductOptionDetailsSchema = z.object({
4
+ title: z.string().min(1),
5
+ values: z.array(z.string()).min(1, "At least one value is required"),
6
+ })
7
+
8
+ export type CreateProductOptionSchema = z.infer<
9
+ typeof CreateProductOptionSchema
10
+ >
11
+ export const CreateProductOptionSchema = z
12
+ .object({
13
+ value_ranks: z.record(z.string(), z.number()).optional(),
14
+ })
15
+ .merge(CreateProductOptionDetailsSchema)
@@ -0,0 +1 @@
1
+ export { ProductOptionCreate as Component } from "./product-option-create"
@@ -0,0 +1,10 @@
1
+ import { RouteFocusModal } from "../../../components/modals"
2
+ import { CreateProductOptionForm } from "./components/create-product-option-form/create-product-option-form"
3
+
4
+ export const ProductOptionCreate = () => {
5
+ return (
6
+ <RouteFocusModal>
7
+ <CreateProductOptionForm />
8
+ </RouteFocusModal>
9
+ )
10
+ }
@@ -0,0 +1,14 @@
1
+ import { HttpTypes } from "@medusajs/types"
2
+ import { UIMatch } from "react-router-dom"
3
+
4
+ export const ProductOptionBreadcrumb = (
5
+ match: UIMatch<HttpTypes.AdminProductOptionResponse>
6
+ ) => {
7
+ const productOption = match.data.product_option
8
+
9
+ if (!productOption) {
10
+ return null
11
+ }
12
+
13
+ return <span>{productOption.title}</span>
14
+ }
@@ -0,0 +1 @@
1
+ export * from "./product-option-general-section.tsx"
@@ -0,0 +1,84 @@
1
+ import { PencilSquare, Trash } from "@medusajs/icons"
2
+ import { HttpTypes } from "@medusajs/types"
3
+ import { Badge, Container, Heading } from "@medusajs/ui"
4
+ import { useMemo } from "react"
5
+ import { useTranslation } from "react-i18next"
6
+ import { ActionMenu } from "../../../../../components/common/action-menu"
7
+ import { useDeleteProductOptionAction } from "../../../common/hooks/use-delete-product-option-action.tsx"
8
+
9
+ export const ProductOptionGeneralSection = ({
10
+ productOption,
11
+ }: {
12
+ productOption: HttpTypes.AdminProductOption
13
+ }) => {
14
+ const { t } = useTranslation()
15
+
16
+ const handleDelete = useDeleteProductOptionAction(productOption)
17
+
18
+ const sortedValues = useMemo(() => {
19
+ if (!productOption.values) {
20
+ return []
21
+ }
22
+
23
+ return [...productOption.values].sort((a: any, b: any) => {
24
+ const rankA = a.rank ?? Number.MAX_VALUE
25
+ const rankB = b.rank ?? Number.MAX_VALUE
26
+ return rankA - rankB
27
+ })
28
+ }, [productOption.values])
29
+
30
+ return (
31
+ <Container className="divide-y p-0">
32
+ <div className="flex items-center justify-between px-6 py-4">
33
+ <Heading>{productOption.title}</Heading>
34
+ <div className="flex items-center gap-x-2">
35
+ <Badge
36
+ size="xsmall"
37
+ color={productOption.is_exclusive ? "grey" : "blue"}
38
+ >
39
+ {t(
40
+ `general.${productOption.is_exclusive ? "exclusive" : "global"}`
41
+ )}
42
+ </Badge>
43
+ <ActionMenu
44
+ groups={[
45
+ {
46
+ actions: [
47
+ {
48
+ label: t("actions.edit"),
49
+ icon: <PencilSquare />,
50
+ to: "edit",
51
+ },
52
+ ],
53
+ },
54
+ {
55
+ actions: [
56
+ {
57
+ label: t("actions.delete"),
58
+ icon: <Trash />,
59
+ onClick: handleDelete,
60
+ },
61
+ ],
62
+ },
63
+ ]}
64
+ />
65
+ </div>
66
+ </div>
67
+ <div className="px-6 py-4">
68
+ <ValuesDisplay values={sortedValues} />
69
+ </div>
70
+ </Container>
71
+ )
72
+ }
73
+
74
+ const ValuesDisplay = ({ values }: { values: any[] }) => {
75
+ return (
76
+ <div className="flex flex-wrap items-center gap-2">
77
+ {values.map((value) => (
78
+ <Badge key={value.id} size="xsmall">
79
+ {value.value}
80
+ </Badge>
81
+ ))}
82
+ </div>
83
+ )
84
+ }
@@ -0,0 +1 @@
1
+ export { ProductOptionProductSection } from "./product-option-product-section"
@@ -0,0 +1,82 @@
1
+ import { Container } from "@medusajs/ui"
2
+ import { keepPreviousData } from "@tanstack/react-query"
3
+ import { useMemo } from "react"
4
+ import { useTranslation } from "react-i18next"
5
+ import { DataTable } from "../../../../../components/data-table"
6
+ import { useProducts } from "../../../../../hooks/api"
7
+ import { useProductTableColumns } from "../../../../../hooks/table/columns"
8
+ import { useProductTableQuery } from "../../../../../hooks/table/query"
9
+
10
+ type ProductOptionProductSectionProps = {
11
+ productOptionId: string
12
+ }
13
+
14
+ const PAGE_SIZE = 10
15
+
16
+ export const ProductOptionProductSection = ({
17
+ productOptionId,
18
+ }: ProductOptionProductSectionProps) => {
19
+ const { t } = useTranslation()
20
+
21
+ const { searchParams } = useProductTableQuery({ pageSize: PAGE_SIZE })
22
+
23
+ const { products, count, isLoading, isError, error } = useProducts(
24
+ {
25
+ limit: PAGE_SIZE,
26
+ ...searchParams,
27
+ fields:
28
+ "id,title,handle,thumbnail,status,*collection,*sales_channels,*product_options,variants.id",
29
+ option_id: productOptionId,
30
+ },
31
+ {
32
+ placeholderData: keepPreviousData,
33
+ enabled: !!productOptionId,
34
+ }
35
+ )
36
+
37
+ const columns = useColumns()
38
+
39
+ if (isError) {
40
+ throw error
41
+ }
42
+
43
+ return (
44
+ <Container className="divide-y p-0">
45
+ <DataTable
46
+ data={products}
47
+ columns={columns}
48
+ rowCount={count}
49
+ pageSize={PAGE_SIZE}
50
+ getRowId={(row) => row.id}
51
+ heading={t("products.domain")}
52
+ emptyState={{
53
+ empty: {
54
+ heading: (
55
+ <span className="text-ui-fg-subtle">
56
+ {t("general.noRecordsMessage")}
57
+ </span>
58
+ ),
59
+ description: (
60
+ <span className="text-ui-fg-muted">
61
+ {t("productOptions.products.list.noRecords")}
62
+ </span>
63
+ ),
64
+ },
65
+ filtered: {
66
+ heading: t("general.noRecordsMessage"),
67
+ description: t("general.noRecordsMessageFiltered"),
68
+ },
69
+ }}
70
+ isLoading={isLoading}
71
+ enableSearch={true}
72
+ rowHref={(row) => `/products/${row.id}`}
73
+ />
74
+ </Container>
75
+ )
76
+ }
77
+
78
+ const useColumns = () => {
79
+ const columns = useProductTableColumns()
80
+
81
+ return useMemo(() => [...columns], [columns])
82
+ }
@@ -0,0 +1,3 @@
1
+ export { ProductOptionBreadcrumb as Breadcrumb } from "./breadcrumb"
2
+ export { ProductOptionDetail as Component } from "./product-option-detail"
3
+ export { productOptionLoader as loader } from "./loader"
@@ -0,0 +1,18 @@
1
+ import { LoaderFunctionArgs } from "react-router-dom"
2
+
3
+ import { sdk } from "../../../lib/client"
4
+ import { queryClient } from "../../../lib/query-client"
5
+ import { productOptionsQueryKeys } from "../../../hooks/api/product-options.tsx"
6
+
7
+ const productOptionDetailQuery = (id: string) => ({
8
+ queryKey: productOptionsQueryKeys.detail(id),
9
+ queryFn: async () =>
10
+ sdk.admin.productOption.retrieve(id, { fields: "-products" }),
11
+ })
12
+
13
+ export const productOptionLoader = async ({ params }: LoaderFunctionArgs) => {
14
+ const id = params.id
15
+ const query = productOptionDetailQuery(id!)
16
+
17
+ return queryClient.ensureQueryData(query)
18
+ }
@@ -0,0 +1,50 @@
1
+ import { useLoaderData, useParams } from "react-router-dom"
2
+
3
+ import { SingleColumnPageSkeleton } from "../../../components/common/skeleton"
4
+ import { SingleColumnPage } from "../../../components/layout/pages"
5
+ import { useExtension } from "../../../providers/extension-provider"
6
+ import { useProductOption } from "../../../hooks/api"
7
+ import { productOptionLoader } from "./loader.ts"
8
+ import { ProductOptionGeneralSection } from "./components/product-option-general-section"
9
+ import { ProductOptionProductSection } from "./components/product-option-product-section"
10
+
11
+ export const ProductOptionDetail = () => {
12
+ const { id } = useParams()
13
+
14
+ const initialData = useLoaderData() as Awaited<
15
+ ReturnType<typeof productOptionLoader>
16
+ >
17
+
18
+ const { getWidgets } = useExtension()
19
+
20
+ const { product_option, isLoading, isError, error } = useProductOption(
21
+ id!,
22
+ { fields: "-products" },
23
+ {
24
+ initialData,
25
+ }
26
+ )
27
+
28
+ if (isLoading || !product_option) {
29
+ return <SingleColumnPageSkeleton sections={2} showJSON showMetadata />
30
+ }
31
+
32
+ if (isError) {
33
+ throw error
34
+ }
35
+
36
+ return (
37
+ <SingleColumnPage
38
+ widgets={{
39
+ after: getWidgets("product_option.details.after"),
40
+ before: getWidgets("product_option.details.before"),
41
+ }}
42
+ showJSON
43
+ showMetadata
44
+ data={product_option}
45
+ >
46
+ <ProductOptionGeneralSection productOption={product_option} />
47
+ <ProductOptionProductSection productOptionId={product_option.id} />
48
+ </SingleColumnPage>
49
+ )
50
+ }
@@ -0,0 +1,94 @@
1
+ import { Input } from "@medusajs/ui"
2
+ import { useEffect } from "react"
3
+ import { UseFormReturn, useWatch } from "react-hook-form"
4
+ import { useTranslation } from "react-i18next"
5
+
6
+ import { Form } from "../../../../../components/common/form"
7
+ import { ChipInput } from "../../../../../components/inputs/chip-input"
8
+ import { EditProductOptionSchema } from "./schema"
9
+
10
+ type EditProductOptionDetailsProps = {
11
+ form: UseFormReturn<EditProductOptionSchema>
12
+ }
13
+
14
+ export const EditProductOptionDetails = ({
15
+ form,
16
+ }: EditProductOptionDetailsProps) => {
17
+ const { t } = useTranslation()
18
+
19
+ const values = useWatch({
20
+ control: form.control,
21
+ name: "values",
22
+ })
23
+
24
+ const valueRanks = useWatch({
25
+ control: form.control,
26
+ name: "value_ranks",
27
+ })
28
+
29
+ useEffect(() => {
30
+ if (!values || !valueRanks) {
31
+ return
32
+ }
33
+
34
+ const validValueSet = new Set(values)
35
+ const currentRanks = { ...valueRanks }
36
+ let hasStaleEntries = false
37
+
38
+ Object.keys(currentRanks).forEach((key) => {
39
+ if (!validValueSet.has(key)) {
40
+ delete currentRanks[key]
41
+ hasStaleEntries = true
42
+ }
43
+ })
44
+
45
+ if (hasStaleEntries) {
46
+ form.setValue("value_ranks", currentRanks, {
47
+ shouldDirty: true,
48
+ shouldTouch: true,
49
+ })
50
+ }
51
+ }, [values, valueRanks, form])
52
+
53
+ return (
54
+ <div className="flex flex-col gap-y-4">
55
+ <Form.Field
56
+ control={form.control}
57
+ name="title"
58
+ render={({ field }) => {
59
+ return (
60
+ <Form.Item>
61
+ <Form.Label>{t("productOptions.fields.title.label")}</Form.Label>
62
+ <Form.Control>
63
+ <Input
64
+ autoComplete="off"
65
+ {...field}
66
+ placeholder={t("productOptions.fields.title.placeholder")}
67
+ />
68
+ </Form.Control>
69
+ <Form.ErrorMessage />
70
+ </Form.Item>
71
+ )
72
+ }}
73
+ />
74
+ <Form.Field
75
+ control={form.control}
76
+ name="values"
77
+ render={({ field }) => {
78
+ return (
79
+ <Form.Item>
80
+ <Form.Label>{t("productOptions.fields.values.label")}</Form.Label>
81
+ <Form.Control>
82
+ <ChipInput
83
+ {...field}
84
+ placeholder={t("productOptions.fields.values.placeholder")}
85
+ />
86
+ </Form.Control>
87
+ <Form.ErrorMessage />
88
+ </Form.Item>
89
+ )
90
+ }}
91
+ />
92
+ </div>
93
+ )
94
+ }
@@ -0,0 +1,116 @@
1
+ import { zodResolver } from "@hookform/resolvers/zod"
2
+ import { HttpTypes } from "@medusajs/types"
3
+ import { Button, toast } from "@medusajs/ui"
4
+ import { useMemo } from "react"
5
+ import { useForm } from "react-hook-form"
6
+ import { useTranslation } from "react-i18next"
7
+
8
+ import { RouteDrawer, useRouteModal } from "../../../../../components/modals"
9
+ import { KeyboundForm } from "../../../../../components/utilities/keybound-form"
10
+ import { useUpdateProductOption } from "../../../../../hooks/api"
11
+ import { EditProductOptionDetails } from "./edit-product-option-details"
12
+ import { EditProductOptionOrganize } from "./edit-product-option-organize"
13
+ import { EditProductOptionSchema } from "./schema"
14
+
15
+ type EditProductOptionFormProps = {
16
+ productOption: HttpTypes.AdminProductOption
17
+ }
18
+
19
+ export const EditProductOptionForm = ({
20
+ productOption,
21
+ }: EditProductOptionFormProps) => {
22
+ const { t } = useTranslation()
23
+ const { handleSuccess } = useRouteModal()
24
+
25
+ const { sortedValues, existingRanks } = useMemo(() => {
26
+ if (!productOption.values) {
27
+ return { sortedValues: [], existingRanks: {} }
28
+ }
29
+
30
+ const ranks: Record<string, number> = {}
31
+ productOption.values.forEach((v: any) => {
32
+ if (v.rank !== undefined && v.rank !== null) {
33
+ ranks[v.value] = v.rank
34
+ }
35
+ })
36
+
37
+ const sorted = [...productOption.values].sort((a: any, b: any) => {
38
+ const rankA = a.rank ?? Number.MAX_VALUE
39
+ const rankB = b.rank ?? Number.MAX_VALUE
40
+ return rankA - rankB
41
+ })
42
+
43
+ return {
44
+ sortedValues: sorted.map((v: any) => v.value),
45
+ existingRanks: ranks,
46
+ }
47
+ }, [productOption.values])
48
+
49
+ const form = useForm<EditProductOptionSchema>({
50
+ defaultValues: {
51
+ title: productOption.title,
52
+ values: sortedValues,
53
+ value_ranks: existingRanks,
54
+ },
55
+ resolver: zodResolver(EditProductOptionSchema),
56
+ })
57
+
58
+ const { mutateAsync, isPending } = useUpdateProductOption(productOption.id)
59
+
60
+ const handleSubmit = form.handleSubmit((data) => {
61
+ const { title, values, value_ranks } = data
62
+
63
+ mutateAsync(
64
+ {
65
+ title,
66
+ values,
67
+ ranks: value_ranks,
68
+ },
69
+ {
70
+ onSuccess: ({ product_option }) => {
71
+ toast.success(
72
+ t("productOptions.edit.successToast", {
73
+ title: product_option.title,
74
+ })
75
+ )
76
+
77
+ handleSuccess()
78
+ },
79
+ onError: (error) => {
80
+ toast.error(error.message)
81
+ },
82
+ }
83
+ )
84
+ })
85
+
86
+ return (
87
+ <RouteDrawer.Form form={form}>
88
+ <KeyboundForm
89
+ onSubmit={handleSubmit}
90
+ className="flex size-full flex-col overflow-hidden"
91
+ >
92
+ <RouteDrawer.Body className="flex flex-col gap-y-4 overflow-auto p-4">
93
+ <EditProductOptionDetails form={form} />
94
+ <EditProductOptionOrganize form={form} />
95
+ </RouteDrawer.Body>
96
+ <RouteDrawer.Footer>
97
+ <div className="flex items-center justify-end gap-x-2">
98
+ <RouteDrawer.Close asChild>
99
+ <Button size="small" variant="secondary">
100
+ {t("actions.cancel")}
101
+ </Button>
102
+ </RouteDrawer.Close>
103
+ <Button
104
+ size="small"
105
+ variant="primary"
106
+ type="submit"
107
+ isLoading={isPending}
108
+ >
109
+ {t("actions.save")}
110
+ </Button>
111
+ </div>
112
+ </RouteDrawer.Footer>
113
+ </KeyboundForm>
114
+ </RouteDrawer.Form>
115
+ )
116
+ }
@@ -0,0 +1,77 @@
1
+ import { Text } from "@medusajs/ui"
2
+ import { useMemo } from "react"
3
+ import { UseFormReturn, useWatch } from "react-hook-form"
4
+ import { SortableList } from "../../../../../components/common/sortable-list"
5
+ import { EditProductOptionSchema } from "./schema"
6
+
7
+ type EditProductOptionOrganizeProps = {
8
+ form: UseFormReturn<EditProductOptionSchema>
9
+ }
10
+
11
+ type ValueItem = {
12
+ id: string
13
+ value: string
14
+ rank: number
15
+ }
16
+
17
+ export const EditProductOptionOrganize = ({
18
+ form,
19
+ }: EditProductOptionOrganizeProps) => {
20
+ const values = useWatch({
21
+ control: form.control,
22
+ name: "values",
23
+ })
24
+
25
+ const valueRanks = useWatch({
26
+ control: form.control,
27
+ name: "value_ranks",
28
+ })
29
+
30
+ const items = useMemo<ValueItem[]>(() => {
31
+ if (!values || values.length === 0) {
32
+ return []
33
+ }
34
+
35
+ return values
36
+ .map((value, index) => ({
37
+ id: value,
38
+ value,
39
+ rank: valueRanks?.[value] ?? index,
40
+ }))
41
+ .sort((a, b) => a.rank - b.rank)
42
+ }, [values, valueRanks])
43
+
44
+ const handleChange = (newItems: ValueItem[]) => {
45
+ const newRanks: Record<string, number> = {}
46
+ newItems.forEach((item, index) => {
47
+ newRanks[item.value] = index
48
+ })
49
+
50
+ form.setValue("value_ranks", newRanks, {
51
+ shouldDirty: true,
52
+ shouldTouch: true,
53
+ })
54
+ }
55
+
56
+ if (!values || values.length === 0) {
57
+ return null
58
+ }
59
+
60
+ return (
61
+ <SortableList
62
+ items={items}
63
+ onChange={handleChange}
64
+ renderItem={(item) => (
65
+ <SortableList.Item
66
+ id={item.id}
67
+ className="border-ui-border-base border-b last:border-b-0"
68
+ >
69
+ <div className="flex flex-1 items-center gap-x-3 px-4 py-3">
70
+ <SortableList.DragHandle />
71
+ <Text size="small">{item.value}</Text>
72
+ </div>
73
+ </SortableList.Item>
74
+ )}
75
+ />
76
+ )
77
+ }
@@ -0,0 +1 @@
1
+ export { EditProductOptionForm } from "./edit-product-option-form"