@medusajs/dashboard 3.0.0-snapshot-20251216135612 → 3.0.0-snapshot-20260102125810

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 (291) 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 +39 -3
  9. package/dist/app.js +18590 -16625
  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-2DULKOPN.mjs → chunk-5ISRTMYH.mjs} +1 -1
  25. package/dist/{chunk-PHLCT2HA.mjs → chunk-5ZPKY5XP.mjs} +1 -1
  26. package/dist/{chunk-U6G4M5LP.mjs → chunk-6EXKKRZY.mjs} +1 -1
  27. package/dist/{chunk-GRZSG4EP.mjs → chunk-74WPAWMW.mjs} +127 -226
  28. package/dist/chunk-7AXHHXCX.mjs +4 -0
  29. package/dist/{chunk-YKYVCQRS.mjs → chunk-BJ44US4R.mjs} +382 -242
  30. package/dist/{chunk-BMS2QLJY.mjs → chunk-BMJ5RYIO.mjs} +1 -1
  31. package/dist/{chunk-Y2YVTIJI.mjs → chunk-CW5PQUJ6.mjs} +1 -1
  32. package/dist/{chunk-IAV7IKJ6.mjs → chunk-DN5BRKVN.mjs} +1 -1
  33. package/dist/{chunk-RS7DWLEP.mjs → chunk-DQUXK4WW.mjs} +1 -1
  34. package/dist/{chunk-2XTBDCGE.mjs → chunk-DWH2OQFK.mjs} +1 -1
  35. package/dist/{chunk-CCQD65EY.mjs → chunk-FKNW5MLZ.mjs} +360 -288
  36. package/dist/{chunk-HTCYX4VD.mjs → chunk-G35UUG2P.mjs} +1 -1
  37. package/dist/{chunk-QKALAT7P.mjs → chunk-GGONMC2N.mjs} +1 -1
  38. package/dist/{chunk-FYWHE3W5.mjs → chunk-GJN5SXGZ.mjs} +1 -1
  39. package/dist/{chunk-CVHJAKLQ.mjs → chunk-HDO2UCKF.mjs} +1 -1
  40. package/dist/{chunk-3C6WQ7NH.mjs → chunk-IWNNQ6HA.mjs} +37 -3
  41. package/dist/{chunk-OL24RDYM.mjs → chunk-KHJL6MUQ.mjs} +8 -8
  42. package/dist/{chunk-6P4Q4AAP.mjs → chunk-KPI4WFJU.mjs} +3 -3
  43. package/dist/{chunk-AHZLMCZF.mjs → chunk-KSDXSKJ7.mjs} +1 -1
  44. package/dist/{chunk-BZKI5J2M.mjs → chunk-L7A2JIQY.mjs} +3 -3
  45. package/dist/{chunk-XY7A7GZJ.mjs → chunk-LEAMWI5H.mjs} +1 -1
  46. package/dist/{chunk-A4XYK3MY.mjs → chunk-LG5B2BVB.mjs} +2 -2
  47. package/dist/{chunk-23GTCEOV.mjs → chunk-ML7BA2NY.mjs} +1 -1
  48. package/dist/{chunk-OFN7DIZA.mjs → chunk-OYHFHBNL.mjs} +94 -7
  49. package/dist/{chunk-OSHH5GAS.mjs → chunk-Q34TRFIW.mjs} +1 -13
  50. package/dist/{chunk-UJ2TMPV4.mjs → chunk-QBIQEVTU.mjs} +0 -12
  51. package/dist/{chunk-YYOPBKME.mjs → chunk-QHRAWZGA.mjs} +3 -3
  52. package/dist/{chunk-A2WBKOXJ.mjs → chunk-SO2DE6MM.mjs} +2 -2
  53. package/dist/chunk-T6DDBLGY.mjs +13 -0
  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 +85 -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-EFIRUBRO.mjs → product-attributes-JTPGHUXZ.mjs} +13 -13
  116. package/dist/{product-create-K6EWZHIT.mjs → product-create-VFLLQWVX.mjs} +485 -561
  117. package/dist/{product-create-variant-ERKHTEJZ.mjs → product-create-variant-PLZ5Q2R6.mjs} +14 -14
  118. package/dist/{product-detail-DKPZDEIY.mjs → product-detail-ISHFZV7X.mjs} +45 -66
  119. package/dist/{product-edit-55YXTIGO.mjs → product-edit-6EDDNIM6.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-JUGLFR2K.mjs +336 -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-4SZDF5H5.mjs +325 -0
  132. package/dist/{product-organization-N3VBRXF4.mjs → product-organization-OEUSIWNQ.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-HUNKY7CO.mjs → translations-edit-6P3KC4MO.mjs} +285 -320
  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 +304 -12
  219. package/src/i18n/translations/en.json +85 -5
  220. package/src/i18n/translations/es.json +21 -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 +192 -0
  230. package/src/routes/product-options/product-option-create/components/create-product-option-form/create-product-option-organize.tsx +77 -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 +296 -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 +253 -41
  272. package/src/routes/translations/translation-list/translation-list.tsx +8 -2
  273. package/src/routes/translations/translations-edit/components/translations-edit-form/translations-edit-form.tsx +337 -380
  274. package/src/routes/translations/translations-edit/translations-edit.tsx +8 -13
  275. package/dist/chunk-HGRIOEAR.mjs +0 -32
  276. package/dist/chunk-XDJ7OMBR.mjs +0 -160
  277. package/dist/login-VNOLI5YG.mjs +0 -201
  278. package/dist/product-create-option-7AOXAA4S.mjs +0 -151
  279. package/dist/product-edit-option-LWJT3CYJ.mjs +0 -156
  280. package/dist/translation-list-S5Z6PG2R.mjs +0 -295
  281. package/src/routes/products/product-create-option/components/create-product-option-form/create-product-option-form.tsx +0 -122
  282. package/src/routes/products/product-create-option/components/create-product-option-form/index.ts +0 -1
  283. package/src/routes/products/product-create-option/index.ts +0 -1
  284. package/src/routes/products/product-create-option/product-create-option.tsx +0 -30
  285. package/src/routes/products/product-edit-option/components/edit-product-option-form/edit-product-option-form.tsx +0 -123
  286. package/src/routes/products/product-edit-option/components/edit-product-option-form/index.ts +0 -1
  287. package/src/routes/products/product-edit-option/index.ts +0 -1
  288. package/src/routes/products/product-edit-option/product-edit-option.tsx +0 -35
  289. package/dist/{chunk-FBYTX6K7.mjs → chunk-ORWRXJJT.mjs} +4 -4
  290. package/dist/{location-list-KVBA6J47.mjs → location-list-3WP65J3E.mjs} +1 -1
  291. package/dist/{user-list-YTZQNYSO.mjs → user-list-3EAKK4XC.mjs} +3 -3
@@ -0,0 +1,192 @@
1
+ import { zodResolver } from "@hookform/resolvers/zod"
2
+ import { Button, ProgressStatus, ProgressTabs, toast } from "@medusajs/ui"
3
+ import { useForm } from "react-hook-form"
4
+ import { useTranslation } from "react-i18next"
5
+
6
+ import { useState } from "react"
7
+ import {
8
+ RouteFocusModal,
9
+ useRouteModal,
10
+ } from "../../../../../components/modals"
11
+ import { KeyboundForm } from "../../../../../components/utilities/keybound-form"
12
+ import { useCreateProductOption } from "../../../../../hooks/api"
13
+ import { CreateProductOptionDetails } from "./create-product-option-details"
14
+ import { CreateProductOptionOrganize } from "./create-product-option-organize"
15
+ import {
16
+ CreateProductOptionDetailsSchema,
17
+ CreateProductOptionSchema,
18
+ } from "./schema"
19
+ import { useDocumentDirection } from "../../../../../hooks/use-document-direction"
20
+
21
+ enum Tab {
22
+ DETAILS = "details",
23
+ ORGANIZE = "organize",
24
+ }
25
+
26
+ export const CreateProductOptionForm = () => {
27
+ const { t } = useTranslation()
28
+ const { handleSuccess } = useRouteModal()
29
+ const direction = useDocumentDirection()
30
+ const [activeTab, setActiveTab] = useState<Tab>(Tab.DETAILS)
31
+ const [validDetails, setValidDetails] = useState(false)
32
+
33
+ const form = useForm<CreateProductOptionSchema>({
34
+ defaultValues: {
35
+ title: "",
36
+ values: [],
37
+ value_ranks: {},
38
+ },
39
+ resolver: zodResolver(CreateProductOptionSchema),
40
+ })
41
+
42
+ const handleTabChange = (tab: Tab) => {
43
+ if (tab === Tab.ORGANIZE) {
44
+ const { title, values } = form.getValues()
45
+
46
+ const result = CreateProductOptionDetailsSchema.safeParse({
47
+ title,
48
+ values,
49
+ })
50
+
51
+ if (!result.success) {
52
+ result.error.errors.forEach((error) => {
53
+ form.setError(
54
+ error.path.join(".") as keyof CreateProductOptionSchema,
55
+ {
56
+ type: "manual",
57
+ message: error.message,
58
+ }
59
+ )
60
+ })
61
+
62
+ return
63
+ }
64
+
65
+ form.clearErrors()
66
+ setValidDetails(true)
67
+ }
68
+
69
+ setActiveTab(tab)
70
+ }
71
+
72
+ const { mutateAsync, isPending } = useCreateProductOption()
73
+
74
+ const handleSubmit = form.handleSubmit((data) => {
75
+ const { title, values, value_ranks } = data
76
+
77
+ mutateAsync(
78
+ {
79
+ title,
80
+ values,
81
+ ranks: value_ranks,
82
+ },
83
+ {
84
+ onSuccess: ({ product_option }) => {
85
+ toast.success(
86
+ t("productOptions.create.successToast", {
87
+ title: product_option.title,
88
+ })
89
+ )
90
+
91
+ handleSuccess(`/product-options/${product_option.id}`)
92
+ },
93
+ onError: (error) => {
94
+ toast.error(error.message)
95
+ },
96
+ }
97
+ )
98
+ })
99
+
100
+ const organizeStatus: ProgressStatus =
101
+ form.getFieldState("value_ranks")?.isDirty || activeTab === Tab.ORGANIZE
102
+ ? "in-progress"
103
+ : "not-started"
104
+
105
+ const detailsStatus: ProgressStatus = validDetails
106
+ ? "completed"
107
+ : "in-progress"
108
+
109
+ return (
110
+ <RouteFocusModal.Form form={form}>
111
+ <KeyboundForm
112
+ onSubmit={handleSubmit}
113
+ className="flex size-full flex-col overflow-hidden"
114
+ >
115
+ <ProgressTabs
116
+ dir={direction}
117
+ value={activeTab}
118
+ onValueChange={(tab) => handleTabChange(tab as Tab)}
119
+ className="flex size-full flex-col"
120
+ >
121
+ <RouteFocusModal.Header>
122
+ <div className="flex w-full items-center justify-between">
123
+ <div className="-my-2 w-full max-w-[400px] border-l">
124
+ <ProgressTabs.List className="grid w-full grid-cols-2">
125
+ <ProgressTabs.Trigger
126
+ value={Tab.DETAILS}
127
+ status={detailsStatus}
128
+ className="w-full min-w-0 overflow-hidden"
129
+ >
130
+ <span className="truncate">
131
+ {t("productOptions.create.tabs.details")}
132
+ </span>
133
+ </ProgressTabs.Trigger>
134
+ <ProgressTabs.Trigger
135
+ value={Tab.ORGANIZE}
136
+ status={organizeStatus}
137
+ className="w-full min-w-0 overflow-hidden"
138
+ >
139
+ <span className="truncate">
140
+ {t("productOptions.create.tabs.organize")}
141
+ </span>
142
+ </ProgressTabs.Trigger>
143
+ </ProgressTabs.List>
144
+ </div>
145
+ </div>
146
+ </RouteFocusModal.Header>
147
+ <RouteFocusModal.Body className="flex size-full flex-col overflow-auto">
148
+ <ProgressTabs.Content value={Tab.DETAILS}>
149
+ <CreateProductOptionDetails form={form} />
150
+ </ProgressTabs.Content>
151
+ <ProgressTabs.Content
152
+ value={Tab.ORGANIZE}
153
+ className="bg-ui-bg-subtle flex-1"
154
+ >
155
+ <CreateProductOptionOrganize form={form} />
156
+ </ProgressTabs.Content>
157
+ </RouteFocusModal.Body>
158
+ <RouteFocusModal.Footer>
159
+ <div className="flex items-center justify-end gap-x-2">
160
+ <RouteFocusModal.Close asChild>
161
+ <Button size="small" variant="secondary">
162
+ {t("actions.cancel")}
163
+ </Button>
164
+ </RouteFocusModal.Close>
165
+ {activeTab === Tab.ORGANIZE ? (
166
+ <Button
167
+ key="submit-btn"
168
+ size="small"
169
+ variant="primary"
170
+ type="submit"
171
+ isLoading={isPending}
172
+ >
173
+ {t("actions.save")}
174
+ </Button>
175
+ ) : (
176
+ <Button
177
+ key="continue-btn"
178
+ size="small"
179
+ variant="primary"
180
+ type="button"
181
+ onClick={() => handleTabChange(Tab.ORGANIZE)}
182
+ >
183
+ {t("actions.continue")}
184
+ </Button>
185
+ )}
186
+ </div>
187
+ </RouteFocusModal.Footer>
188
+ </ProgressTabs>
189
+ </KeyboundForm>
190
+ </RouteFocusModal.Form>
191
+ )
192
+ }
@@ -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 { CreateProductOptionSchema } from "./schema"
6
+
7
+ type CreateProductOptionOrganizeProps = {
8
+ form: UseFormReturn<CreateProductOptionSchema>
9
+ }
10
+
11
+ type ValueItem = {
12
+ id: string
13
+ value: string
14
+ rank: number
15
+ }
16
+
17
+ export const CreateProductOptionOrganize = ({
18
+ form,
19
+ }: CreateProductOptionOrganizeProps) => {
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 { 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
+ }