@medusajs/dashboard 2.12.3-preview-20251217120154 → 2.12.3-preview-20251217123836

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 (276) hide show
  1. package/dist/{add-campaign-promotions-4QLXO46G.mjs → add-campaign-promotions-OYPGISTF.mjs} +2 -2
  2. package/dist/add-locales-GGNZCABB.mjs +81 -0
  3. package/dist/{adjust-inventory-2MFA656V.mjs → adjust-inventory-26YXAXQL.mjs} +4 -3
  4. package/dist/{api-key-management-create-ZUVF3SLR.mjs → api-key-management-create-EMP32G2D.mjs} +6 -5
  5. package/dist/{api-key-management-detail-THD74G7I.mjs → api-key-management-detail-FRUN2KFK.mjs} +11 -10
  6. package/dist/{api-key-management-edit-42HNUSXL.mjs → api-key-management-edit-IBM3ZXHK.mjs} +6 -5
  7. package/dist/{api-key-management-list-66CFOILY.mjs → api-key-management-list-KC5GOWAU.mjs} +4 -3
  8. package/dist/{api-key-management-sales-channels-HN3THM5J.mjs → api-key-management-sales-channels-LUB5G6RC.mjs} +6 -5
  9. package/dist/app.css +72 -0
  10. package/dist/app.js +13657 -11324
  11. package/dist/app.mjs +12 -11
  12. package/dist/{campaign-budget-edit-6DMZGUCI.mjs → campaign-budget-edit-B5MSE26J.mjs} +2 -2
  13. package/dist/{campaign-configuration-UNFRB5HR.mjs → campaign-configuration-QOE3D2BE.mjs} +2 -2
  14. package/dist/{campaign-create-QXG7PD3S.mjs → campaign-create-FJOECKGT.mjs} +3 -3
  15. package/dist/{campaign-detail-WR3XHHV2.mjs → campaign-detail-HM3GQJLQ.mjs} +11 -10
  16. package/dist/{campaign-edit-HN4US7DV.mjs → campaign-edit-K2POQH7M.mjs} +2 -2
  17. package/dist/{categories-metadata-BCZ463KN.mjs → categories-metadata-WKL3MGD7.mjs} +14 -13
  18. package/dist/{category-create-FFIBWXAH.mjs → category-create-KHJZSC7G.mjs} +5 -4
  19. package/dist/{category-detail-4HXJ5Z7N.mjs → category-detail-UTWWDKFP.mjs} +29 -13
  20. package/dist/{category-edit-74MZ6TXN.mjs → category-edit-CTA2EPDG.mjs} +5 -4
  21. package/dist/{category-list-OCJY2UJV.mjs → category-list-QBYJ4T3R.mjs} +20 -4
  22. package/dist/{category-organize-B6EFBFKB.mjs → category-organize-SXP33XET.mjs} +5 -4
  23. package/dist/{category-products-2HXLZDHL.mjs → category-products-XXBTCXFF.mjs} +14 -13
  24. package/dist/{chunk-XIVBRONM.mjs → chunk-23GTCEOV.mjs} +1 -1
  25. package/dist/{chunk-2NZDHAOE.mjs → chunk-27MGH3HR.mjs} +2 -2
  26. package/dist/{chunk-N3FCYZV6.mjs → chunk-2XTBDCGE.mjs} +1 -1
  27. package/dist/{chunk-RU4ZS47V.mjs → chunk-4JQR6QNW.mjs} +2 -2
  28. package/dist/{chunk-F65T2X7G.mjs → chunk-5F427YCP.mjs} +5 -5
  29. package/dist/{chunk-PNPT4W2F.mjs → chunk-5ISRTMYH.mjs} +1 -1
  30. package/dist/{chunk-Z6NHG5LQ.mjs → chunk-6P4Q4AAP.mjs} +3 -3
  31. package/dist/{chunk-F5NZDW7L.mjs → chunk-A2WBKOXJ.mjs} +2 -2
  32. package/dist/{chunk-F63F3FIT.mjs → chunk-A4XYK3MY.mjs} +2 -2
  33. package/dist/{chunk-NT6C7CHT.mjs → chunk-AWRCV3ME.mjs} +1 -1
  34. package/dist/{chunk-5X6SKB2P.mjs → chunk-BMS2QLJY.mjs} +1 -1
  35. package/dist/{chunk-3YPRDSZ6.mjs → chunk-BZKI5J2M.mjs} +1 -1
  36. package/dist/{chunk-25UWYEJX.mjs → chunk-CQOOXWPZ.mjs} +1 -1
  37. package/dist/{chunk-42OOM4DJ.mjs → chunk-D6UW7URG.mjs} +23 -6
  38. package/dist/{chunk-4YAPXPLM.mjs → chunk-DQUXK4WW.mjs} +1 -1
  39. package/dist/{chunk-MY6I7UJG.mjs → chunk-DTCIBQO2.mjs} +3 -3
  40. package/dist/{chunk-W7SLPIO2.mjs → chunk-FKNW5MLZ.mjs} +676 -151
  41. package/dist/{chunk-N4O5FAUC.mjs → chunk-FYWHE3W5.mjs} +1 -1
  42. package/dist/{chunk-QPKW37WR.mjs → chunk-GLBHPDR4.mjs} +249 -14
  43. package/dist/chunk-HGRIOEAR.mjs +32 -0
  44. package/dist/{chunk-PXJFCRIV.mjs → chunk-HNJ65IND.mjs} +4 -4
  45. package/dist/{chunk-Y5UO73CH.mjs → chunk-HTCYX4VD.mjs} +1 -1
  46. package/dist/{chunk-34KFHPN6.mjs → chunk-IAV7IKJ6.mjs} +1 -1
  47. package/dist/{store-add-locales-JLRTXG3Q.mjs → chunk-IKTGFXWR.mjs} +3 -67
  48. package/dist/{chunk-6W3BKVOC.mjs → chunk-KFYQTOGB.mjs} +1 -1
  49. package/dist/{chunk-EQTBJSBZ.mjs → chunk-KIIT4BNH.mjs} +3 -0
  50. package/dist/{chunk-LHNU4DYZ.mjs → chunk-KSDXSKJ7.mjs} +2 -2
  51. package/dist/{chunk-D4J3CEWH.mjs → chunk-N3SAXQVR.mjs} +2 -2
  52. package/dist/{chunk-Q6MSICBU.mjs → chunk-O333RR6K.mjs} +1 -1
  53. package/dist/{chunk-YAPTGHGP.mjs → chunk-OK6NZN2A.mjs} +1 -1
  54. package/dist/{chunk-7AX6R6G6.mjs → chunk-OSHH5GAS.mjs} +13 -1
  55. package/dist/{chunk-35LSPN2U.mjs → chunk-QKALAT7P.mjs} +1 -1
  56. package/dist/{chunk-UM4OIJZ3.mjs → chunk-S22SJRPO.mjs} +1 -1
  57. package/dist/{chunk-HFT7Q5Y5.mjs → chunk-SG2JZPTG.mjs} +1 -1
  58. package/dist/{chunk-B3AOQW2B.mjs → chunk-U6G4M5LP.mjs} +1 -1
  59. package/dist/{chunk-HWKLNKOY.mjs → chunk-U726TGCM.mjs} +1 -1
  60. package/dist/{chunk-HKIF5HVL.mjs → chunk-UJ2TMPV4.mjs} +12 -0
  61. package/dist/{chunk-5DUMSVP6.mjs → chunk-UMCJYHAD.mjs} +3 -3
  62. package/dist/{chunk-NHDFPGQ3.mjs → chunk-UWY5ZV66.mjs} +13 -1
  63. package/dist/{chunk-7ZYDO3XO.mjs → chunk-VFF5WB7C.mjs} +299 -194
  64. package/dist/{chunk-XXBP2Z5M.mjs → chunk-WVA4O7QS.mjs} +1 -1
  65. package/dist/{chunk-AH4DORIW.mjs → chunk-XY7A7GZJ.mjs} +1 -1
  66. package/dist/{chunk-FQDPOKEK.mjs → chunk-Y2YVTIJI.mjs} +1 -1
  67. package/dist/{chunk-L4NOVTH5.mjs → chunk-YIOBBZUB.mjs} +2 -2
  68. package/dist/{chunk-LEJUZW3P.mjs → chunk-Z6BFNHEO.mjs} +12 -0
  69. package/dist/{collection-add-products-JWAQ277T.mjs → collection-add-products-42F7H77E.mjs} +14 -13
  70. package/dist/{collection-create-DCRW344Z.mjs → collection-create-GWKWVT7B.mjs} +5 -4
  71. package/dist/{collection-detail-F4ON763T.mjs → collection-detail-PXIS3G64.mjs} +28 -12
  72. package/dist/{collection-edit-UNFEW6HT.mjs → collection-edit-EZIO2BR5.mjs} +5 -4
  73. package/dist/{collection-list-4C5HVO6Y.mjs → collection-list-O74CGY24.mjs} +29 -13
  74. package/dist/{collection-metadata-P4A5DXBQ.mjs → collection-metadata-U6FMA4IC.mjs} +14 -13
  75. package/dist/{customer-create-XCLZENJS.mjs → customer-create-IA56MXE6.mjs} +2 -2
  76. package/dist/{customer-create-address-4HZUIAM7.mjs → customer-create-address-27M25HTO.mjs} +2 -2
  77. package/dist/{customer-detail-X3RQJPXP.mjs → customer-detail-OMTFJ6CE.mjs} +12 -11
  78. package/dist/{customer-edit-BW5GVTQI.mjs → customer-edit-XQ5NWIVX.mjs} +2 -2
  79. package/dist/{customer-group-add-customers-SOXMSWQH.mjs → customer-group-add-customers-XMR2WBXX.mjs} +2 -2
  80. package/dist/{customer-group-create-6YV7W6TI.mjs → customer-group-create-RY4FVEIW.mjs} +2 -2
  81. package/dist/{customer-group-detail-NG32WKJT.mjs → customer-group-detail-ADK3M5LG.mjs} +11 -10
  82. package/dist/{customer-group-edit-7NXU34BQ.mjs → customer-group-edit-MQWARIQZ.mjs} +2 -2
  83. package/dist/{customer-group-list-EBETHKKO.mjs → customer-group-list-7ZRQ2HWU.mjs} +11 -10
  84. package/dist/{customer-group-metadata-WK2BGEGO.mjs → customer-group-metadata-JQJRLLXG.mjs} +3 -3
  85. package/dist/{customer-metadata-ZUWCUTM7.mjs → customer-metadata-HN2DYD5I.mjs} +3 -3
  86. package/dist/{customers-add-customer-group-7FYD54MZ.mjs → customers-add-customer-group-5U27WHJB.mjs} +13 -12
  87. package/dist/{edit-inventory-item-Y5W6VTHJ.mjs → edit-inventory-item-H7DAZWIT.mjs} +4 -3
  88. package/dist/{edit-inventory-item-attributes-NAAZDFEJ.mjs → edit-inventory-item-attributes-7HTXXPGZ.mjs} +4 -3
  89. package/dist/{edit-reservation-2UD4RZJX.mjs → edit-reservation-OVTRZHJR.mjs} +5 -4
  90. package/dist/{edit-rules-2W3BXEXR.mjs → edit-rules-BM2ERGVJ.mjs} +13 -12
  91. package/dist/en.json +38 -0
  92. package/dist/{inventory-create-VQEK7C5T.mjs → inventory-create-7MA7B5N2.mjs} +14 -13
  93. package/dist/{inventory-detail-HUHQ6ZNE.mjs → inventory-detail-B4PRHZK3.mjs} +11 -10
  94. package/dist/{inventory-list-RUSSPKHE.mjs → inventory-list-RXJPSVZE.mjs} +2 -1
  95. package/dist/{inventory-metadata-X52VGHA7.mjs → inventory-metadata-C7MJ3GY5.mjs} +14 -13
  96. package/dist/{inventory-stock-EKTOYN2F.mjs → inventory-stock-WVTYPJTX.mjs} +15 -14
  97. package/dist/{invite-VKHAAF4Z.mjs → invite-FVE4ZBKB.mjs} +2 -2
  98. package/dist/{location-create-RPLEXUZR.mjs → location-create-WZ43K3W7.mjs} +2 -2
  99. package/dist/{location-detail-UVWYE6Q2.mjs → location-detail-KO6EBDK5.mjs} +12 -11
  100. package/dist/{location-edit-L4ISEQHX.mjs → location-edit-VCS7GVWQ.mjs} +2 -2
  101. package/dist/{location-fulfillment-providers-HZZUHPVL.mjs → location-fulfillment-providers-IORBE3E3.mjs} +15 -14
  102. package/dist/{location-list-Z42ZYHUF.mjs → location-list-KVBA6J47.mjs} +2 -2
  103. package/dist/{location-sales-channels-XIMOK66B.mjs → location-sales-channels-P3QJTFDT.mjs} +5 -4
  104. package/dist/{location-service-zone-create-7ZNVLEE3.mjs → location-service-zone-create-J43WN6G4.mjs} +3 -3
  105. package/dist/{location-service-zone-edit-6OUAFEZD.mjs → location-service-zone-edit-KHHMSPXD.mjs} +2 -2
  106. package/dist/{location-service-zone-manage-areas-EOULKRYD.mjs → location-service-zone-manage-areas-6ZPMKMSX.mjs} +3 -3
  107. package/dist/{location-service-zone-shipping-option-create-M7G4SKNP.mjs → location-service-zone-shipping-option-create-2R3ZFLVK.mjs} +15 -14
  108. package/dist/{location-service-zone-shipping-option-edit-M2HJAASG.mjs → location-service-zone-shipping-option-edit-4CGPQ3VT.mjs} +2 -2
  109. package/dist/{location-service-zone-shipping-option-pricing-23QLNSBT.mjs → location-service-zone-shipping-option-pricing-5HN2Z5RB.mjs} +4 -4
  110. package/dist/{login-NSCAMAU6.mjs → login-XKB6OR7I.mjs} +13 -12
  111. package/dist/{manage-locations-JD6FSBOT.mjs → manage-locations-7HH6R4UP.mjs} +4 -3
  112. package/dist/{order-allocate-items-5T57NZOB.mjs → order-allocate-items-HZGGYJ42.mjs} +6 -5
  113. package/dist/{order-create-claim-TE56IE7P.mjs → order-create-claim-NKCOGF4A.mjs} +15 -14
  114. package/dist/{order-create-edit-RDFYBVFY.mjs → order-create-edit-UNQYXGLL.mjs} +14 -13
  115. package/dist/{order-create-exchange-B5ZOEQCJ.mjs → order-create-exchange-WI7OA2WO.mjs} +15 -14
  116. package/dist/{order-create-fulfillment-SRRP6X64.mjs → order-create-fulfillment-2LJTEWDY.mjs} +13 -12
  117. package/dist/{order-create-refund-PTOM65QK.mjs → order-create-refund-7K6UJXGP.mjs} +13 -12
  118. package/dist/{order-create-return-3C2QVNYN.mjs → order-create-return-52GHGW5Z.mjs} +7 -6
  119. package/dist/{order-create-shipment-HNILNFXB.mjs → order-create-shipment-ZTDLLUBY.mjs} +13 -12
  120. package/dist/{order-detail-3OUZOQ7G.mjs → order-detail-JTRUMRLO.mjs} +15 -14
  121. package/dist/{order-edit-billing-address-3IO7GXIK.mjs → order-edit-billing-address-YHYNVLOE.mjs} +13 -12
  122. package/dist/{order-edit-email-QIAEOUBV.mjs → order-edit-email-TCQPEVZY.mjs} +13 -12
  123. package/dist/{order-edit-shipping-address-QGGMHGBC.mjs → order-edit-shipping-address-CFSYQLKD.mjs} +13 -12
  124. package/dist/{order-export-JYYFTOWR.mjs → order-export-G4SBNEJ7.mjs} +14 -13
  125. package/dist/{order-list-RFR3HNDQ.mjs → order-list-GRLQWN4L.mjs} +6 -5
  126. package/dist/{order-metadata-ZBDTMBHZ.mjs → order-metadata-KGPB37VL.mjs} +14 -13
  127. package/dist/{order-receive-return-V42FTCFY.mjs → order-receive-return-JER24SEV.mjs} +14 -13
  128. package/dist/{order-request-transfer-A6NNIN2O.mjs → order-request-transfer-3FBUYZNT.mjs} +13 -12
  129. package/dist/{price-list-configuration-RTUEJENE.mjs → price-list-configuration-6S3MLNXQ.mjs} +6 -5
  130. package/dist/{price-list-create-GRT27KOM.mjs → price-list-create-CXZCFFTP.mjs} +17 -16
  131. package/dist/{price-list-detail-VKIWM7HP.mjs → price-list-detail-XOMU6U5J.mjs} +13 -12
  132. package/dist/{price-list-edit-SW7YIR3Z.mjs → price-list-edit-53UW35L3.mjs} +5 -4
  133. package/dist/{price-list-list-LF2PL4B5.mjs → price-list-list-DG5YEZ44.mjs} +4 -3
  134. package/dist/{price-list-prices-add-ISLZDQNI.mjs → price-list-prices-add-SDX5CQME.mjs} +16 -15
  135. package/dist/{price-list-prices-edit-C6SX2TLJ.mjs → price-list-prices-edit-EKB6NI5D.mjs} +7 -6
  136. package/dist/{product-attributes-REJ5NF2D.mjs → product-attributes-MXDPSOWM.mjs} +14 -13
  137. package/dist/{product-create-4TFE3JHQ.mjs → product-create-3O34JJLS.mjs} +16 -15
  138. package/dist/{product-create-option-GEGQNSMQ.mjs → product-create-option-7AOXAA4S.mjs} +4 -3
  139. package/dist/{product-create-variant-WI6D5CDG.mjs → product-create-variant-OTJKT6WI.mjs} +14 -13
  140. package/dist/{product-detail-BEIDGPKY.mjs → product-detail-SYTLG5D3.mjs} +59 -17
  141. package/dist/{product-edit-ZRQUK5OQ.mjs → product-edit-W72S22NM.mjs} +14 -13
  142. package/dist/{product-edit-option-HBIOBQTS.mjs → product-edit-option-LWJT3CYJ.mjs} +4 -3
  143. package/dist/{product-export-GIEFG4MA.mjs → product-export-57UUAGXF.mjs} +15 -14
  144. package/dist/{product-image-variants-edit-AAMRK4KQ.mjs → product-image-variants-edit-2BW5BJON.mjs} +13 -12
  145. package/dist/{product-import-43UXKCL3.mjs → product-import-6EM4VUXP.mjs} +13 -12
  146. package/dist/{product-list-D4CVUBQQ.mjs → product-list-5V5GEH5K.mjs} +25 -12
  147. package/dist/{product-media-EGK66SRX.mjs → product-media-3VJ7KENL.mjs} +4 -3
  148. package/dist/{product-metadata-55CAQTPS.mjs → product-metadata-JZLHBLZQ.mjs} +14 -13
  149. package/dist/{product-organization-7J7MUM6F.mjs → product-organization-SVXTCWIF.mjs} +15 -14
  150. package/dist/{product-prices-WQDE4XYT.mjs → product-prices-5ZL2RP7A.mjs} +5 -4
  151. package/dist/{product-sales-channels-GIJ7Q57M.mjs → product-sales-channels-PPXUG4KT.mjs} +5 -4
  152. package/dist/{product-shipping-profile-KIDRTPWA.mjs → product-shipping-profile-ETQFZ7DC.mjs} +4 -3
  153. package/dist/{product-stock-22S2HGMM.mjs → product-stock-SJJABF6I.mjs} +15 -14
  154. package/dist/{product-tag-create-7GPE246H.mjs → product-tag-create-XXO4AQEC.mjs} +13 -12
  155. package/dist/{product-tag-detail-R6C2TTA4.mjs → product-tag-detail-BSK64HXL.mjs} +30 -14
  156. package/dist/{product-tag-edit-WA6FGSIP.mjs → product-tag-edit-ENCGDT7E.mjs} +13 -12
  157. package/dist/{product-tag-list-U7PLHMLF.mjs → product-tag-list-SLQGCNDZ.mjs} +30 -14
  158. package/dist/{product-tag-metadata-EJ7FEXSE.mjs → product-tag-metadata-EPXHMU2K.mjs} +14 -13
  159. package/dist/{product-type-create-HBDROTGB.mjs → product-type-create-DRFXTL5O.mjs} +4 -3
  160. package/dist/{product-type-detail-TYQ7MJFE.mjs → product-type-detail-4CRRU7YK.mjs} +29 -13
  161. package/dist/{product-type-edit-H3JCMVRV.mjs → product-type-edit-SRHCZDK7.mjs} +4 -3
  162. package/dist/{product-type-list-KIKWBPD7.mjs → product-type-list-QQKAHBJ3.mjs} +19 -3
  163. package/dist/{product-type-metadata-Q6W5GL7Z.mjs → product-type-metadata-73OKOGPP.mjs} +14 -13
  164. package/dist/{product-variant-detail-SPJ7MZTK.mjs → product-variant-detail-RPHLG4HU.mjs} +27 -11
  165. package/dist/{product-variant-edit-BW7QMHC5.mjs → product-variant-edit-JF7NN64Y.mjs} +13 -12
  166. package/dist/{product-variant-manage-inventory-items-BLHCH6QD.mjs → product-variant-manage-inventory-items-Y2VEOHP7.mjs} +4 -3
  167. package/dist/{product-variant-media-KLSUISGH.mjs → product-variant-media-2WLVNGI4.mjs} +4 -3
  168. package/dist/{product-variant-metadata-GDMVC4MZ.mjs → product-variant-metadata-HU2CXGPO.mjs} +14 -13
  169. package/dist/{profile-edit-YBBXXCCK.mjs → profile-edit-ZNXO6WME.mjs} +2 -2
  170. package/dist/{promotion-add-campaign-JRTM7WBR.mjs → promotion-add-campaign-DO67QK6M.mjs} +3 -3
  171. package/dist/{promotion-create-HF6DTKCC.mjs → promotion-create-BHA3FQG2.mjs} +15 -14
  172. package/dist/{promotion-detail-HTM3F2VO.mjs → promotion-detail-F3QSR52W.mjs} +11 -10
  173. package/dist/{promotion-edit-details-AREFQ6O5.mjs → promotion-edit-details-6BSOOWQN.mjs} +2 -2
  174. package/dist/{refund-reason-create-RKIH46D6.mjs → refund-reason-create-ZA5TKW2Z.mjs} +13 -12
  175. package/dist/{refund-reason-edit-V3KC6NMO.mjs → refund-reason-edit-N2CRCLKZ.mjs} +13 -12
  176. package/dist/{refund-reason-list-M5BOHFWU.mjs → refund-reason-list-SE4TMGMT.mjs} +11 -10
  177. package/dist/{region-add-countries-OF5RUSYU.mjs → region-add-countries-2VAVXMJQ.mjs} +2 -2
  178. package/dist/{region-create-BGOWVCTR.mjs → region-create-NA7Y2LN4.mjs} +2 -2
  179. package/dist/{region-edit-5GO25GDF.mjs → region-edit-WAU347DP.mjs} +2 -2
  180. package/dist/{region-metadata-5NGUNAXZ.mjs → region-metadata-O5NZBWXP.mjs} +14 -13
  181. package/dist/{reservation-create-Z2SUMBGT.mjs → reservation-create-ZCIYM6JI.mjs} +5 -4
  182. package/dist/{reservation-detail-JH77TQHN.mjs → reservation-detail-UFK6XIXE.mjs} +11 -10
  183. package/dist/{reservation-list-WHBGZSVH.mjs → reservation-list-B47DXTA7.mjs} +3 -2
  184. package/dist/{reservation-metadata-ERCWXYVL.mjs → reservation-metadata-AEJEKGLV.mjs} +14 -13
  185. package/dist/{reset-password-5TAYQEUV.mjs → reset-password-Y5WVXV4Q.mjs} +2 -2
  186. package/dist/{return-reason-create-NXWDC6AC.mjs → return-reason-create-HM54WRJY.mjs} +2 -2
  187. package/dist/{return-reason-edit-M6IKNNKB.mjs → return-reason-edit-VSIRHVQ6.mjs} +2 -2
  188. package/dist/{sales-channel-add-products-DTAXWL6H.mjs → sales-channel-add-products-2LMB7EF5.mjs} +14 -13
  189. package/dist/{sales-channel-create-E7BDTPWD.mjs → sales-channel-create-MI7HHZYE.mjs} +5 -4
  190. package/dist/{sales-channel-detail-XWHV7J7X.mjs → sales-channel-detail-EUQ4STQI.mjs} +12 -11
  191. package/dist/{sales-channel-edit-HMPA3Y3L.mjs → sales-channel-edit-VSHOIR37.mjs} +5 -4
  192. package/dist/{sales-channel-list-7NSEFCXQ.mjs → sales-channel-list-JXKGHX4G.mjs} +11 -10
  193. package/dist/{sales-channel-metadata-CMZN74K7.mjs → sales-channel-metadata-AJMQ5SQ2.mjs} +14 -13
  194. package/dist/{shipping-option-type-create-4BN2GBHP.mjs → shipping-option-type-create-YVVIA2XC.mjs} +13 -12
  195. package/dist/{shipping-option-type-detail-YB3TCED2.mjs → shipping-option-type-detail-ZZW36XLK.mjs} +12 -11
  196. package/dist/{shipping-option-type-edit-MDO7ALAU.mjs → shipping-option-type-edit-O6F74T3A.mjs} +13 -12
  197. package/dist/{shipping-option-type-list-DJAT4RKX.mjs → shipping-option-type-list-SPTE7MT6.mjs} +12 -11
  198. package/dist/{shipping-profile-create-752TMBBE.mjs → shipping-profile-create-26R64I3Z.mjs} +2 -2
  199. package/dist/{shipping-profile-metadata-HQI7PCPU.mjs → shipping-profile-metadata-7WFE55VG.mjs} +14 -13
  200. package/dist/{store-add-currencies-PEWNEWCH.mjs → store-add-currencies-OX2WXFMS.mjs} +2 -2
  201. package/dist/store-add-locales-GWCGIXHU.mjs +81 -0
  202. package/dist/{store-detail-MOLAYNQZ.mjs → store-detail-YLJLBBZE.mjs} +11 -10
  203. package/dist/{store-edit-FTLPT55E.mjs → store-edit-5ZS562ZO.mjs} +2 -2
  204. package/dist/{store-metadata-ZJD23O4D.mjs → store-metadata-BZ57I2E6.mjs} +14 -13
  205. package/dist/{tax-region-create-NCO2OQ3I.mjs → tax-region-create-FGTV7VJL.mjs} +13 -12
  206. package/dist/{tax-region-detail-RIABXU7D.mjs → tax-region-detail-PPIMD7OX.mjs} +21 -20
  207. package/dist/{tax-region-edit-YZALLSZY.mjs → tax-region-edit-ELZKA7YH.mjs} +13 -12
  208. package/dist/{tax-region-list-BXZBUXRJ.mjs → tax-region-list-P4LDOPZD.mjs} +3 -3
  209. package/dist/{tax-region-province-create-5JP3PVID.mjs → tax-region-province-create-GKJ7WMRS.mjs} +2 -2
  210. package/dist/{tax-region-province-detail-UKLRVJS2.mjs → tax-region-province-detail-FV2NDT3E.mjs} +20 -19
  211. package/dist/{tax-region-tax-override-create-33XGHH7A.mjs → tax-region-tax-override-create-N572MQPZ.mjs} +16 -15
  212. package/dist/{tax-region-tax-override-edit-WOKZSPTY.mjs → tax-region-tax-override-edit-5DCSJW6D.mjs} +17 -16
  213. package/dist/{tax-region-tax-rate-create-O2SDSXCH.mjs → tax-region-tax-rate-create-MECFGVPL.mjs} +2 -2
  214. package/dist/{tax-region-tax-rate-edit-W62VEVKT.mjs → tax-region-tax-rate-edit-7DYSPYOD.mjs} +2 -2
  215. package/dist/translation-list-FK7XYLHX.mjs +527 -0
  216. package/dist/translations-edit-VRXZI5KW.mjs +678 -0
  217. package/dist/{user-detail-KEDJ7N3Q.mjs → user-detail-BJUXLZZQ.mjs} +2 -1
  218. package/dist/{user-edit-LXQCTDGA.mjs → user-edit-ZBB4RMTP.mjs} +2 -2
  219. package/dist/{user-invite-YH33WUC2.mjs → user-invite-GAGIM5DO.mjs} +2 -2
  220. package/dist/{user-metadata-4WOICEE7.mjs → user-metadata-GRJZZ524.mjs} +14 -13
  221. package/dist/{workflow-execution-detail-OKM6P4SX.mjs → workflow-execution-detail-HXTFWGKG.mjs} +11 -10
  222. package/package.json +9 -9
  223. package/src/components/common/icon-avatar/icon-avatar.tsx +4 -0
  224. package/src/components/data-grid/components/data-grid-cell-container.tsx +13 -4
  225. package/src/components/data-grid/components/data-grid-multiline-cell.tsx +103 -0
  226. package/src/components/data-grid/components/data-grid-readonly-cell.tsx +13 -3
  227. package/src/components/data-grid/components/data-grid-root.tsx +271 -60
  228. package/src/components/data-grid/components/data-grid-text-cell.tsx +7 -7
  229. package/src/components/data-grid/components/data-grid-textarea-modal-cell.tsx +233 -0
  230. package/src/components/data-grid/components/index.ts +2 -0
  231. package/src/components/data-grid/data-grid.tsx +16 -1
  232. package/src/components/data-grid/helpers/create-data-grid-column-helper.ts +18 -0
  233. package/src/components/data-grid/hooks/use-data-grid-cell.tsx +13 -1
  234. package/src/components/data-grid/hooks/use-data-grid-form-handlers.tsx +1 -0
  235. package/src/components/data-grid/hooks/use-data-grid-keydown-event.tsx +65 -7
  236. package/src/components/data-grid/types.ts +1 -0
  237. package/src/components/layout/settings-layout/settings-layout.tsx +11 -1
  238. package/src/components/modals/route-focus-modal/route-focus-modal.tsx +8 -3
  239. package/src/components/modals/route-modal-form/route-modal-form.tsx +6 -1
  240. package/src/components/modals/route-modal-provider/route-provider.tsx +16 -4
  241. package/src/dashboard-app/routes/get-route.map.tsx +23 -0
  242. package/src/hooks/api/categories.tsx +32 -0
  243. package/src/hooks/api/collections.tsx +31 -0
  244. package/src/hooks/api/index.ts +1 -0
  245. package/src/hooks/api/product-types.tsx +32 -0
  246. package/src/hooks/api/product-variants.tsx +47 -3
  247. package/src/hooks/api/products.tsx +29 -0
  248. package/src/hooks/api/tags.tsx +32 -0
  249. package/src/hooks/api/translations.tsx +315 -0
  250. package/src/hooks/use-infinite-list.tsx +92 -0
  251. package/src/i18n/translations/$schema.json +122 -0
  252. package/src/i18n/translations/en.json +38 -0
  253. package/src/i18n/translations/es.json +38 -0
  254. package/src/routes/categories/category-detail/components/category-general-section/category-general-section.tsx +16 -1
  255. package/src/routes/categories/category-list/components/category-list-table/category-list-table.tsx +16 -1
  256. package/src/routes/collections/collection-detail/components/collection-general-section/collection-general-section.tsx +16 -1
  257. package/src/routes/collections/collection-list/components/collection-list-table/collection-row-actions.tsx +16 -1
  258. package/src/routes/product-tags/product-tag-detail/components/product-tag-general-section/product-tag-general-section.tsx +16 -1
  259. package/src/routes/product-tags/product-tag-list/components/product-tag-list-table/product-tag-list-table.tsx +16 -1
  260. package/src/routes/product-types/product-type-detail/components/product-type-general-section/product-type-general-section.tsx +16 -1
  261. package/src/routes/product-types/product-type-list/components/product-type-list-table/product-table-row-actions.tsx +16 -1
  262. package/src/routes/product-variants/product-variant-detail/components/variant-general-section/variant-general-section.tsx +16 -1
  263. package/src/routes/products/product-detail/components/product-general-section/product-general-section.tsx +16 -1
  264. package/src/routes/products/product-detail/components/product-variant-section/product-variant-section.tsx +32 -1
  265. package/src/routes/products/product-list/components/product-list-table/product-list-table.tsx +15 -1
  266. package/src/routes/translations/add-locales/add-locales.tsx +29 -0
  267. package/src/routes/translations/add-locales/index.tsx +1 -0
  268. package/src/routes/translations/translation-list/components/active-locales-section/active-locales-section.tsx +93 -0
  269. package/src/routes/translations/translation-list/components/translation-list-section/translation-list-section.tsx +48 -0
  270. package/src/routes/translations/translation-list/components/translations-completion-section/translations-completion-section.tsx +285 -0
  271. package/src/routes/translations/translation-list/index.tsx +1 -0
  272. package/src/routes/translations/translation-list/translation-list.tsx +138 -0
  273. package/src/routes/translations/translations-edit/components/translations-edit-form/index.ts +1 -0
  274. package/src/routes/translations/translations-edit/components/translations-edit-form/translations-edit-form.tsx +747 -0
  275. package/src/routes/translations/translations-edit/index.ts +1 -0
  276. package/src/routes/translations/translations-edit/translations-edit.tsx +90 -0
@@ -1,8 +1,9 @@
1
- import { PencilSquare, Trash } from "@medusajs/icons"
1
+ import { GlobeEurope, PencilSquare, Trash } from "@medusajs/icons"
2
2
  import { HttpTypes } from "@medusajs/types"
3
3
  import { useTranslation } from "react-i18next"
4
4
  import { ActionMenu } from "../../../../../components/common/action-menu"
5
5
  import { useDeleteProductTypeAction } from "../../../common/hooks/use-delete-product-type-action"
6
+ import { useFeatureFlag } from "../../../../../providers/feature-flag-provider"
6
7
 
7
8
  type ProductTypeRowActionsProps = {
8
9
  productType: HttpTypes.AdminProductType
@@ -16,6 +17,7 @@ export const ProductTypeRowActions = ({
16
17
  productType.id,
17
18
  productType.value
18
19
  )
20
+ const isTranslationsEnabled = useFeatureFlag("translation")
19
21
 
20
22
  return (
21
23
  <ActionMenu
@@ -29,6 +31,19 @@ export const ProductTypeRowActions = ({
29
31
  },
30
32
  ],
31
33
  },
34
+ ...(isTranslationsEnabled
35
+ ? [
36
+ {
37
+ actions: [
38
+ {
39
+ icon: <GlobeEurope />,
40
+ label: t("translations.actions.manage"),
41
+ to: `/settings/translations/edit?reference=product_type&reference_id=${productType.id}`,
42
+ },
43
+ ],
44
+ },
45
+ ]
46
+ : []),
32
47
  {
33
48
  actions: [
34
49
  {
@@ -1,4 +1,4 @@
1
- import { Component, PencilSquare, Trash } from "@medusajs/icons"
1
+ import { Component, GlobeEurope, PencilSquare, Trash } from "@medusajs/icons"
2
2
  import { HttpTypes } from "@medusajs/types"
3
3
  import { Badge, Container, Heading, usePrompt } from "@medusajs/ui"
4
4
  import { useTranslation } from "react-i18next"
@@ -7,6 +7,7 @@ import { useNavigate } from "react-router-dom"
7
7
  import { ActionMenu } from "../../../../../components/common/action-menu"
8
8
  import { SectionRow } from "../../../../../components/common/section"
9
9
  import { useDeleteVariant } from "../../../../../hooks/api/products"
10
+ import { useFeatureFlag } from "../../../../../providers/feature-flag-provider"
10
11
 
11
12
  type VariantGeneralSectionProps = {
12
13
  variant: HttpTypes.AdminProductVariant
@@ -16,6 +17,7 @@ export function VariantGeneralSection({ variant }: VariantGeneralSectionProps) {
16
17
  const { t } = useTranslation()
17
18
  const prompt = usePrompt()
18
19
  const navigate = useNavigate()
20
+ const isTranslationsEnabled = useFeatureFlag("translation")
19
21
 
20
22
  const hasInventoryKit = variant.inventory?.length > 1
21
23
 
@@ -70,6 +72,19 @@ export function VariantGeneralSection({ variant }: VariantGeneralSectionProps) {
70
72
  },
71
73
  ],
72
74
  },
75
+ ...(isTranslationsEnabled
76
+ ? [
77
+ {
78
+ actions: [
79
+ {
80
+ label: t("translations.actions.manage"),
81
+ to: `/settings/translations/edit?reference=product_variant&reference_id=${variant.id}`,
82
+ icon: <GlobeEurope />,
83
+ },
84
+ ],
85
+ },
86
+ ]
87
+ : []),
73
88
  {
74
89
  actions: [
75
90
  {
@@ -1,4 +1,4 @@
1
- import { PencilSquare, Trash } from "@medusajs/icons"
1
+ import { GlobeEurope, PencilSquare, Trash } from "@medusajs/icons"
2
2
  import { HttpTypes } from "@medusajs/types"
3
3
  import { Container, Heading, StatusBadge, toast, usePrompt } from "@medusajs/ui"
4
4
  import { useTranslation } from "react-i18next"
@@ -8,6 +8,7 @@ import { ActionMenu } from "../../../../../components/common/action-menu"
8
8
  import { SectionRow } from "../../../../../components/common/section"
9
9
  import { useDeleteProduct } from "../../../../../hooks/api/products"
10
10
  import { useExtension } from "../../../../../providers/extension-provider"
11
+ import { useFeatureFlag } from "../../../../../providers/feature-flag-provider"
11
12
 
12
13
  const productStatusColor = (status: string) => {
13
14
  switch (status) {
@@ -35,6 +36,7 @@ export const ProductGeneralSection = ({
35
36
  const prompt = usePrompt()
36
37
  const navigate = useNavigate()
37
38
  const { getDisplays } = useExtension()
39
+ const isTranslationsEnabled = useFeatureFlag("translation")
38
40
 
39
41
  const displays = getDisplays("product", "general")
40
42
 
@@ -85,6 +87,19 @@ export const ProductGeneralSection = ({
85
87
  },
86
88
  ],
87
89
  },
90
+ ...(isTranslationsEnabled
91
+ ? [
92
+ {
93
+ actions: [
94
+ {
95
+ label: t("translations.actions.manage"),
96
+ to: `/settings/translations/edit?reference=product&reference_id=${product.id}`,
97
+ icon: <GlobeEurope />,
98
+ },
99
+ ],
100
+ },
101
+ ]
102
+ : []),
88
103
  {
89
104
  actions: [
90
105
  {
@@ -1,4 +1,10 @@
1
- import { Buildings, Component, PencilSquare, Trash } from "@medusajs/icons"
1
+ import {
2
+ Buildings,
3
+ Component,
4
+ GlobeEurope,
5
+ PencilSquare,
6
+ Trash,
7
+ } from "@medusajs/icons"
2
8
  import { HttpTypes } from "@medusajs/types"
3
9
  import {
4
10
  Badge,
@@ -26,6 +32,7 @@ import {
26
32
  import { useQueryParams } from "../../../../../hooks/use-query-params"
27
33
  import { PRODUCT_VARIANT_IDS_KEY } from "../../../common/constants"
28
34
  import { Thumbnail } from "../../../../../components/common/thumbnail"
35
+ import { useFeatureFlag } from "../../../../../providers/feature-flag-provider"
29
36
 
30
37
  type ProductVariantSectionProps = {
31
38
  product: HttpTypes.AdminProduct
@@ -38,6 +45,7 @@ export const ProductVariantSection = ({
38
45
  product,
39
46
  }: ProductVariantSectionProps) => {
40
47
  const { t } = useTranslation()
48
+ const isTranslationsEnabled = useFeatureFlag("translation")
41
49
 
42
50
  const { q, order, offset, allow_backorder, manage_inventory } =
43
51
  useQueryParams(
@@ -70,6 +78,11 @@ export const ProductVariantSection = ({
70
78
  }
71
79
  )
72
80
 
81
+ const translationParams = new URLSearchParams()
82
+ variants?.forEach((variant) => {
83
+ translationParams.append("reference_id", variant.id)
84
+ })
85
+
73
86
  if (isError) {
74
87
  throw error
75
88
  }
@@ -115,6 +128,15 @@ export const ProductVariantSection = ({
115
128
  to: `stock`,
116
129
  icon: <Buildings />,
117
130
  },
131
+ ...(isTranslationsEnabled
132
+ ? [
133
+ {
134
+ icon: <GlobeEurope />,
135
+ label: t("translations.actions.manage"),
136
+ to: `/settings/translations/edit?reference=product_variant&${translationParams.toString()}`,
137
+ },
138
+ ]
139
+ : []),
118
140
  ],
119
141
  },
120
142
  ],
@@ -225,6 +247,15 @@ const useColumns = (product: HttpTypes.AdminProduct) => {
225
247
  )
226
248
  },
227
249
  },
250
+ {
251
+ icon: <GlobeEurope />,
252
+ label: t("translations.actions.manage"),
253
+ onClick: () => {
254
+ navigate(
255
+ `/settings/translations/edit?reference=product_variant&reference_id=${variant.id}`
256
+ )
257
+ },
258
+ },
228
259
  ]
229
260
 
230
261
  const secondaryActions: DataTableAction<HttpTypes.AdminProductVariant>[] =
@@ -1,4 +1,4 @@
1
- import { PencilSquare, Trash } from "@medusajs/icons"
1
+ import { GlobeEurope, PencilSquare, Trash } from "@medusajs/icons"
2
2
  import { Button, Container, Heading, toast, usePrompt } from "@medusajs/ui"
3
3
  import { keepPreviousData } from "@tanstack/react-query"
4
4
  import { createColumnHelper } from "@tanstack/react-table"
@@ -110,6 +110,7 @@ const ProductActions = ({ product }: { product: HttpTypes.AdminProduct }) => {
110
110
  const { t } = useTranslation()
111
111
  const prompt = usePrompt()
112
112
  const { mutateAsync } = useDeleteProduct(product.id)
113
+ const isTranslationsEnabled = useFeatureFlag("translation")
113
114
 
114
115
  const handleDelete = async () => {
115
116
  const res = await prompt({
@@ -153,6 +154,19 @@ const ProductActions = ({ product }: { product: HttpTypes.AdminProduct }) => {
153
154
  },
154
155
  ],
155
156
  },
157
+ ...(isTranslationsEnabled
158
+ ? [
159
+ {
160
+ actions: [
161
+ {
162
+ icon: <GlobeEurope />,
163
+ label: t("translations.actions.manage"),
164
+ to: `/settings/translations/edit?reference=product&reference_id=${product.id}`,
165
+ },
166
+ ],
167
+ },
168
+ ]
169
+ : []),
156
170
  {
157
171
  actions: [
158
172
  {
@@ -0,0 +1,29 @@
1
+ import { RouteFocusModal } from "../../../components/modals/route-focus-modal"
2
+ import { useStore } from "../../../hooks/api"
3
+ import { useFeatureFlag } from "../../../providers/feature-flag-provider"
4
+ import { useNavigate } from "react-router-dom"
5
+ import { AddLocalesForm } from "../../store/store-add-locales/components/add-locales-form/add-locales-form"
6
+
7
+ export const TranslationsAddLocales = () => {
8
+ const isEnabled = useFeatureFlag("translation")
9
+ const navigate = useNavigate()
10
+
11
+ if (!isEnabled) {
12
+ navigate(-1)
13
+ return null
14
+ }
15
+
16
+ const { store, isPending, isError, error } = useStore()
17
+
18
+ const ready = !!store && !isPending
19
+
20
+ if (isError) {
21
+ throw error
22
+ }
23
+
24
+ return (
25
+ <RouteFocusModal>
26
+ {ready && <AddLocalesForm store={store} />}
27
+ </RouteFocusModal>
28
+ )
29
+ }
@@ -0,0 +1 @@
1
+ export { TranslationsAddLocales as Component } from "./add-locales"
@@ -0,0 +1,93 @@
1
+ import { PencilSquare, Language } from "@medusajs/icons"
2
+ import {
3
+ Container,
4
+ Heading,
5
+ IconButton,
6
+ InlineTip,
7
+ Text,
8
+ Tooltip,
9
+ } from "@medusajs/ui"
10
+ import { useTranslation } from "react-i18next"
11
+ import { IconAvatar } from "../../../../../components/common/icon-avatar"
12
+ import { HttpTypes } from "@medusajs/types"
13
+ import { useCallback, useState } from "react"
14
+ import { useNavigate } from "react-router-dom"
15
+
16
+ type ActiveLocalesSectionProps = {
17
+ locales: HttpTypes.AdminLocale[]
18
+ }
19
+
20
+ export const ActiveLocalesSection = ({
21
+ locales,
22
+ }: ActiveLocalesSectionProps) => {
23
+ const { t } = useTranslation()
24
+ const navigate = useNavigate()
25
+ const [isHovered, setIsHovered] = useState(false)
26
+
27
+ const handleManageLocales = useCallback(() => {
28
+ navigate("/settings/translations/add-locales")
29
+ }, [navigate])
30
+
31
+ const renderLocales = useCallback(() => {
32
+ const maxLocalesToDetail = 2
33
+ if (locales.length <= maxLocalesToDetail) {
34
+ return locales.map((locale) => locale.name).join(", ")
35
+ }
36
+
37
+ return `${locales
38
+ .slice(0, maxLocalesToDetail)
39
+ .map((locale) => locale.name)
40
+ .join(", ")} + ${locales.length - maxLocalesToDetail}`
41
+ }, [locales])
42
+
43
+ const hasLocales = locales.length > 0
44
+
45
+ return (
46
+ <Container className="flex flex-col p-0">
47
+ <div className="flex items-center justify-between px-6 py-4">
48
+ <Heading level="h2">{t("translations.activeLocales.heading")}</Heading>
49
+ <IconButton variant="transparent" onClick={handleManageLocales}>
50
+ <PencilSquare></PencilSquare>
51
+ </IconButton>
52
+ </div>
53
+ <div className="px-1 pb-1">
54
+ {hasLocales ? (
55
+ <Tooltip
56
+ open={isHovered}
57
+ content={
58
+ <div className="flex flex-col gap-y-1 p-1">
59
+ {locales.map((locale) => (
60
+ <Text key={locale.code} size="small" weight="plus">
61
+ {locale.name}
62
+ </Text>
63
+ ))}
64
+ </div>
65
+ }
66
+ >
67
+ <Container
68
+ className="bg-ui-bg-component border-r-1 flex items-center gap-x-4 px-[19px] py-2"
69
+ onMouseEnter={() => setIsHovered(true)}
70
+ onMouseLeave={() => setIsHovered(false)}
71
+ >
72
+ <IconAvatar className="border-ui-border-base border">
73
+ <Language />
74
+ </IconAvatar>
75
+ <div className="flex flex-col">
76
+ <Text size="small" weight="plus">
77
+ {t("translations.activeLocales.subtitle")}
78
+ </Text>
79
+ <Text className="text-ui-fg-subtle" size="small">
80
+ {renderLocales()}
81
+ </Text>
82
+ </div>
83
+ </Container>
84
+ </Tooltip>
85
+ ) : (
86
+ <InlineTip label="Tip">
87
+ {t("translations.activeLocales.noLocalesTip")}
88
+ </InlineTip>
89
+ )}
90
+ </div>
91
+ </Container>
92
+ )
93
+ }
@@ -0,0 +1,48 @@
1
+ import { Button, Container, Text } from "@medusajs/ui"
2
+ import { Link } from "react-router-dom"
3
+ import { TranslatableEntity } from "../../translation-list"
4
+ import { useTranslation } from "react-i18next"
5
+
6
+ type TranslationListSectionProps = {
7
+ entities: TranslatableEntity[]
8
+ hasLocales: boolean
9
+ }
10
+
11
+ export const TranslationListSection = ({
12
+ entities,
13
+ hasLocales = false,
14
+ }: TranslationListSectionProps) => {
15
+ const { t } = useTranslation()
16
+
17
+ return (
18
+ <Container className="divide-y p-0">
19
+ {entities.map((entity) => (
20
+ <div
21
+ key={entity.reference}
22
+ className="grid grid-cols-[250px_1fr_auto] items-center gap-x-4 px-6 py-4"
23
+ >
24
+ <Text size="small" weight="plus">
25
+ {entity.label}
26
+ </Text>
27
+ <Text size="small" className="text-ui-fg-subtle">
28
+ {t("translations.list.metrics", {
29
+ translated: (entity.translatedCount ?? 0).toLocaleString(),
30
+ total: (entity.totalCount ?? 0).toLocaleString(),
31
+ })}
32
+ </Text>
33
+ <Link
34
+ to={`/settings/translations/edit?reference=${entity.reference}`}
35
+ >
36
+ <Button
37
+ variant="secondary"
38
+ size="small"
39
+ disabled={!hasLocales || !entity.totalCount}
40
+ >
41
+ Edit
42
+ </Button>
43
+ </Link>
44
+ </div>
45
+ ))}
46
+ </Container>
47
+ )
48
+ }
@@ -0,0 +1,285 @@
1
+ import { AdminTranslationEntityStatistics, HttpTypes } from "@medusajs/types"
2
+ import { Container, Divider, Heading, Text, Tooltip } from "@medusajs/ui"
3
+ import { useMemo, useState } from "react"
4
+ import { useTranslation } from "react-i18next"
5
+
6
+ type TranslationsCompletionSectionProps = {
7
+ statistics: Record<string, AdminTranslationEntityStatistics>
8
+ locales: HttpTypes.AdminLocale[]
9
+ }
10
+
11
+ type LocaleStats = {
12
+ code: string
13
+ name: string
14
+ translated: number
15
+ toTranslate: number
16
+ total: number
17
+ }
18
+
19
+ export const TranslationsCompletionSection = ({
20
+ statistics,
21
+ locales,
22
+ }: TranslationsCompletionSectionProps) => {
23
+ const { t } = useTranslation()
24
+ const [hoveredLocale, setHoveredLocale] = useState<string | null>(null)
25
+
26
+ const { translatedCount, totalCount } = Object.values(statistics).reduce(
27
+ (acc, curr) => ({
28
+ translatedCount: acc.translatedCount + curr.translated,
29
+ totalCount: acc.totalCount + curr.expected,
30
+ }),
31
+ { totalCount: 0, translatedCount: 0 }
32
+ )
33
+
34
+ const percentage = totalCount > 0 ? (translatedCount / totalCount) * 100 : 0
35
+ const remaining = Math.max(0, totalCount - translatedCount)
36
+
37
+ const localeStats = useMemo((): LocaleStats[] => {
38
+ const localeMap = new Map<
39
+ string,
40
+ { translated: number; expected: number }
41
+ >()
42
+
43
+ locales.forEach((locale) => {
44
+ localeMap.set(locale.code, { translated: 0, expected: 0 })
45
+ })
46
+
47
+ Object.values(statistics).forEach((entityStats) => {
48
+ if (entityStats.by_locale) {
49
+ Object.entries(entityStats.by_locale).forEach(
50
+ ([localeCode, localeData]) => {
51
+ const existing = localeMap.get(localeCode)
52
+ if (existing) {
53
+ existing.translated += localeData.translated
54
+ existing.expected += localeData.expected
55
+ }
56
+ }
57
+ )
58
+ }
59
+ })
60
+
61
+ return locales.map((locale) => {
62
+ const stats = localeMap.get(locale.code) || { translated: 0, expected: 0 }
63
+ return {
64
+ code: locale.code,
65
+ name: locale.name,
66
+ translated: stats.translated,
67
+ toTranslate: Math.max(0, stats.expected - stats.translated),
68
+ total: stats.expected,
69
+ }
70
+ })
71
+ }, [statistics, locales])
72
+
73
+ const maxTotal = useMemo(
74
+ () => Math.max(...localeStats.map((s) => s.total), 1),
75
+ [localeStats]
76
+ )
77
+
78
+ const localeStatsCount = useMemo(() => localeStats.length, [localeStats])
79
+
80
+ return (
81
+ <Container className="p-0">
82
+ <div className="flex flex-col gap-y-4 px-6 py-4">
83
+ <div className="flex items-center justify-between">
84
+ <Heading level="h2">{t("translations.completion.heading")}</Heading>
85
+ <Text size="small" weight="plus" className="text-ui-fg-subtle">
86
+ {translatedCount.toLocaleString()} {t("general.of")}{" "}
87
+ {totalCount.toLocaleString()}
88
+ </Text>
89
+ </div>
90
+
91
+ <div className="flex h-3 w-full overflow-hidden">
92
+ {percentage > 0 ? (
93
+ <>
94
+ <div
95
+ className="mr-0.5 h-full rounded-sm transition-all"
96
+ style={{
97
+ width: `${percentage}%`,
98
+ backgroundColor: "var(--tag-blue-icon)",
99
+ boxShadow: "inset 0 0 0 0.5px var(--alpha-250)",
100
+ }}
101
+ />
102
+ <div
103
+ className="h-full flex-1 rounded-sm"
104
+ style={{
105
+ backgroundColor: "var(--tag-blue-border)",
106
+ boxShadow: "inset 0 0 0 0.5px var(--alpha-250)",
107
+ }}
108
+ />
109
+ </>
110
+ ) : (
111
+ <div
112
+ className="h-full w-full rounded-sm"
113
+ style={{
114
+ backgroundColor: "var(--tag-blue-border)",
115
+ boxShadow: "inset 0 0 0 0.5px var(--alpha-250)",
116
+ }}
117
+ />
118
+ )}
119
+ </div>
120
+
121
+ <div className="flex items-center justify-between">
122
+ <Text size="small" weight="plus" className="text-ui-fg-subtle">
123
+ {percentage.toFixed(1)}%
124
+ </Text>
125
+ <Text size="small" weight="plus" className="text-ui-fg-subtle">
126
+ {remaining.toLocaleString()} {t("general.remaining").toLowerCase()}
127
+ </Text>
128
+ </div>
129
+ </div>
130
+
131
+ {localeStats.length > 0 && (
132
+ <>
133
+ <Divider variant="dashed" />
134
+ <div className="flex flex-col gap-y-3 px-6 pb-6 pt-4">
135
+ <div className="flex h-32 w-full items-end gap-1">
136
+ {localeStats.map((locale) => {
137
+ const heightPercent = (locale.total / maxTotal) * 100
138
+ const translatedPercent =
139
+ locale.total > 0
140
+ ? (locale.translated / locale.total) * 100
141
+ : 0
142
+
143
+ return (
144
+ <Tooltip
145
+ key={locale.code}
146
+ open={hoveredLocale === locale.code}
147
+ content={
148
+ <div className="flex min-w-[150px] flex-col gap-y-1 p-1">
149
+ <Text size="small" weight="plus">
150
+ {locale.name}
151
+ </Text>
152
+ <div className="flex items-center justify-between">
153
+ <div className="flex items-center gap-x-2">
154
+ <div
155
+ className="h-2 w-2 rounded-full"
156
+ style={{
157
+ backgroundColor: "var(--tag-blue-icon)",
158
+ boxShadow: "inset 0 0 0 0.5px var(--alpha-250)",
159
+ }}
160
+ />
161
+ <Text
162
+ size="small"
163
+ weight="plus"
164
+ className="text-ui-fg-base"
165
+ >
166
+ {t("translations.completion.translated")}
167
+ </Text>
168
+ </div>
169
+ <Text
170
+ size="small"
171
+ weight="plus"
172
+ className="text-ui-fg-base"
173
+ >
174
+ {locale.translated}
175
+ </Text>
176
+ </div>
177
+ <div className="flex items-center justify-between">
178
+ <div className="flex items-center gap-x-2">
179
+ <div
180
+ className="h-2 w-2 rounded-full"
181
+ style={{
182
+ backgroundColor: "var(--tag-blue-border)",
183
+ boxShadow: "inset 0 0 0 0.5px var(--alpha-250)",
184
+ }}
185
+ />
186
+ <Text
187
+ size="small"
188
+ weight="plus"
189
+ className="text-ui-fg-base"
190
+ >
191
+ {t("translations.completion.toTranslate")}
192
+ </Text>
193
+ </div>
194
+ <Text
195
+ size="small"
196
+ weight="plus"
197
+ className="text-ui-fg-base"
198
+ >
199
+ {locale.toTranslate}
200
+ </Text>
201
+ </div>
202
+ </div>
203
+ }
204
+ >
205
+ <div className="flex h-full flex-1 items-end justify-center">
206
+ <div
207
+ className="flex w-full min-w-2 max-w-[96px] flex-col justify-end overflow-hidden rounded-t-sm transition-opacity"
208
+ style={{ height: `${heightPercent}%` }}
209
+ onMouseEnter={() => setHoveredLocale(locale.code)}
210
+ onMouseLeave={() => setHoveredLocale(null)}
211
+ >
212
+ {translatedPercent === 0 ? (
213
+ <div
214
+ className="w-full rounded-sm"
215
+ style={{
216
+ height: "100%",
217
+ backgroundColor: "var(--tag-neutral-bg)",
218
+ boxShadow: "inset 0 0 0 0.5px var(--alpha-250)",
219
+ }}
220
+ />
221
+ ) : (
222
+ <>
223
+ <div
224
+ className="w-full rounded-sm"
225
+ style={{
226
+ height: `${100 - translatedPercent}%`,
227
+ backgroundColor: "var(--tag-blue-border)",
228
+ boxShadow: "inset 0 0 0 0.5px var(--alpha-250)",
229
+ minHeight: locale.toTranslate > 0 ? "2px" : "0",
230
+ }}
231
+ />
232
+ {translatedPercent > 0 && (
233
+ <div
234
+ className="mt-0.5 w-full rounded-sm"
235
+ style={{
236
+ height: `${translatedPercent}%`,
237
+ backgroundColor: "var(--tag-blue-icon)",
238
+ boxShadow:
239
+ "inset 0 0 0 0.5px var(--alpha-250)",
240
+ minHeight:
241
+ locale.translated > 0 ? "2px" : "0",
242
+ }}
243
+ />
244
+ )}
245
+ </>
246
+ )}
247
+ </div>
248
+ </div>
249
+ </Tooltip>
250
+ )
251
+ })}
252
+ </div>
253
+ {localeStatsCount < 9 && (
254
+ <div className="flex w-full gap-1">
255
+ {localeStats.map((locale) => (
256
+ <div
257
+ key={locale.code}
258
+ className="flex flex-1 items-center justify-center"
259
+ >
260
+ <Text
261
+ size="xsmall"
262
+ weight="plus"
263
+ className="text-ui-fg-subtle min-w-2 whitespace-normal break-words text-center leading-tight"
264
+ >
265
+ {localeStatsCount < 6 ? locale.name : locale.code}
266
+ </Text>
267
+ </div>
268
+ ))}
269
+ </div>
270
+ )}
271
+ {localeStatsCount > 9 && (
272
+ <Text
273
+ weight="plus"
274
+ size="xsmall"
275
+ className="text-ui-fg-subtle text-center"
276
+ >
277
+ {t("translations.completion.footer")}
278
+ </Text>
279
+ )}
280
+ </div>
281
+ </>
282
+ )}
283
+ </Container>
284
+ )
285
+ }