@strapi/content-releases 0.0.0-experimental.e3e48deb89bd0a1b6cc69b698696566fa7854a95 → 0.0.0-experimental.e50ef5e2ea57ecf3da5bcf308508b51ee3c0deca

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 (338) hide show
  1. package/LICENSE +17 -1
  2. package/dist/admin/assets/purchase-page-illustration-dark.svg.js +6 -0
  3. package/dist/admin/assets/purchase-page-illustration-dark.svg.js.map +1 -0
  4. package/dist/admin/assets/purchase-page-illustration-dark.svg.mjs +4 -0
  5. package/dist/admin/assets/purchase-page-illustration-dark.svg.mjs.map +1 -0
  6. package/dist/admin/assets/purchase-page-illustration-light.svg.js +6 -0
  7. package/dist/admin/assets/purchase-page-illustration-light.svg.js.map +1 -0
  8. package/dist/admin/assets/purchase-page-illustration-light.svg.mjs +4 -0
  9. package/dist/admin/assets/purchase-page-illustration-light.svg.mjs.map +1 -0
  10. package/dist/admin/components/EntryValidationPopover.js +344 -0
  11. package/dist/admin/components/EntryValidationPopover.js.map +1 -0
  12. package/dist/admin/components/EntryValidationPopover.mjs +342 -0
  13. package/dist/admin/components/EntryValidationPopover.mjs.map +1 -0
  14. package/dist/admin/components/RelativeTime.js +76 -0
  15. package/dist/admin/components/RelativeTime.js.map +1 -0
  16. package/dist/admin/components/RelativeTime.mjs +55 -0
  17. package/dist/admin/components/RelativeTime.mjs.map +1 -0
  18. package/dist/admin/components/ReleaseAction.js +203 -0
  19. package/dist/admin/components/ReleaseAction.js.map +1 -0
  20. package/dist/admin/components/ReleaseAction.mjs +201 -0
  21. package/dist/admin/components/ReleaseAction.mjs.map +1 -0
  22. package/dist/admin/components/ReleaseActionMenu.js +194 -0
  23. package/dist/admin/components/ReleaseActionMenu.js.map +1 -0
  24. package/dist/admin/components/ReleaseActionMenu.mjs +173 -0
  25. package/dist/admin/components/ReleaseActionMenu.mjs.map +1 -0
  26. package/dist/admin/components/ReleaseActionModal.js +268 -0
  27. package/dist/admin/components/ReleaseActionModal.js.map +1 -0
  28. package/dist/admin/components/ReleaseActionModal.mjs +244 -0
  29. package/dist/admin/components/ReleaseActionModal.mjs.map +1 -0
  30. package/dist/admin/components/ReleaseActionOptions.js +104 -0
  31. package/dist/admin/components/ReleaseActionOptions.js.map +1 -0
  32. package/dist/admin/components/ReleaseActionOptions.mjs +102 -0
  33. package/dist/admin/components/ReleaseActionOptions.mjs.map +1 -0
  34. package/dist/admin/components/ReleaseListCell.js +103 -0
  35. package/dist/admin/components/ReleaseListCell.js.map +1 -0
  36. package/dist/admin/components/ReleaseListCell.mjs +100 -0
  37. package/dist/admin/components/ReleaseListCell.mjs.map +1 -0
  38. package/dist/admin/components/ReleaseModal.js +323 -0
  39. package/dist/admin/components/ReleaseModal.js.map +1 -0
  40. package/dist/admin/components/ReleaseModal.mjs +302 -0
  41. package/dist/admin/components/ReleaseModal.mjs.map +1 -0
  42. package/dist/admin/components/ReleasesPanel.js +138 -0
  43. package/dist/admin/components/ReleasesPanel.js.map +1 -0
  44. package/dist/admin/components/ReleasesPanel.mjs +136 -0
  45. package/dist/admin/components/ReleasesPanel.mjs.map +1 -0
  46. package/dist/admin/constants.js +77 -0
  47. package/dist/admin/constants.js.map +1 -0
  48. package/dist/admin/constants.mjs +75 -0
  49. package/dist/admin/constants.mjs.map +1 -0
  50. package/dist/admin/index.js +125 -18
  51. package/dist/admin/index.js.map +1 -1
  52. package/dist/admin/index.mjs +120 -18
  53. package/dist/admin/index.mjs.map +1 -1
  54. package/dist/admin/modules/hooks.js +8 -0
  55. package/dist/admin/modules/hooks.js.map +1 -0
  56. package/dist/admin/modules/hooks.mjs +6 -0
  57. package/dist/admin/modules/hooks.mjs.map +1 -0
  58. package/dist/admin/pages/App.js +29 -0
  59. package/dist/admin/pages/App.js.map +1 -0
  60. package/dist/admin/pages/App.mjs +27 -0
  61. package/dist/admin/pages/App.mjs.map +1 -0
  62. package/dist/admin/pages/PurchaseContentReleases.js +192 -0
  63. package/dist/admin/pages/PurchaseContentReleases.js.map +1 -0
  64. package/dist/admin/pages/PurchaseContentReleases.mjs +190 -0
  65. package/dist/admin/pages/PurchaseContentReleases.mjs.map +1 -0
  66. package/dist/admin/pages/ReleaseDetailsPage.js +772 -0
  67. package/dist/admin/pages/ReleaseDetailsPage.js.map +1 -0
  68. package/dist/admin/pages/ReleaseDetailsPage.mjs +751 -0
  69. package/dist/admin/pages/ReleaseDetailsPage.mjs.map +1 -0
  70. package/dist/admin/pages/ReleasesPage.js +403 -0
  71. package/dist/admin/pages/ReleasesPage.js.map +1 -0
  72. package/dist/admin/pages/ReleasesPage.mjs +381 -0
  73. package/dist/admin/pages/ReleasesPage.mjs.map +1 -0
  74. package/dist/admin/pages/ReleasesSettingsPage.js +225 -0
  75. package/dist/admin/pages/ReleasesSettingsPage.js.map +1 -0
  76. package/dist/admin/pages/ReleasesSettingsPage.mjs +223 -0
  77. package/dist/admin/pages/ReleasesSettingsPage.mjs.map +1 -0
  78. package/dist/admin/pluginId.js +6 -0
  79. package/dist/admin/pluginId.js.map +1 -0
  80. package/dist/admin/pluginId.mjs +4 -0
  81. package/dist/admin/pluginId.mjs.map +1 -0
  82. package/dist/admin/services/release.js +464 -0
  83. package/dist/admin/services/release.js.map +1 -0
  84. package/dist/admin/services/release.mjs +447 -0
  85. package/dist/admin/services/release.mjs.map +1 -0
  86. package/dist/admin/src/components/EntryValidationPopover.d.ts +13 -0
  87. package/dist/admin/src/components/ReleaseAction.d.ts +3 -0
  88. package/dist/admin/src/components/ReleaseActionMenu.d.ts +4 -4
  89. package/dist/admin/src/components/ReleaseActionModal.d.ts +24 -0
  90. package/dist/admin/src/components/ReleaseListCell.d.ts +28 -0
  91. package/dist/admin/src/components/ReleaseModal.d.ts +3 -2
  92. package/dist/admin/src/components/ReleasesPanel.d.ts +3 -0
  93. package/dist/admin/src/constants.d.ts +18 -0
  94. package/dist/admin/src/modules/hooks.d.ts +7 -0
  95. package/dist/admin/src/pages/ReleasesSettingsPage.d.ts +1 -0
  96. package/dist/admin/src/services/release.d.ts +66 -323
  97. package/dist/admin/src/utils/api.d.ts +6 -0
  98. package/dist/admin/src/utils/prefixPluginTranslations.d.ts +3 -0
  99. package/dist/admin/src/utils/time.d.ts +9 -0
  100. package/dist/admin/src/validation/schemas.d.ts +6 -0
  101. package/dist/admin/store/hooks.js +8 -0
  102. package/dist/admin/store/hooks.js.map +1 -0
  103. package/dist/admin/store/hooks.mjs +6 -0
  104. package/dist/admin/store/hooks.mjs.map +1 -0
  105. package/dist/admin/translations/en.json.js +108 -0
  106. package/dist/admin/translations/en.json.js.map +1 -0
  107. package/dist/admin/translations/en.json.mjs +106 -0
  108. package/dist/admin/translations/en.json.mjs.map +1 -0
  109. package/dist/admin/translations/uk.json.js +103 -0
  110. package/dist/admin/translations/uk.json.js.map +1 -0
  111. package/dist/admin/translations/uk.json.mjs +101 -0
  112. package/dist/admin/translations/uk.json.mjs.map +1 -0
  113. package/dist/admin/utils/api.js +8 -0
  114. package/dist/admin/utils/api.js.map +1 -0
  115. package/dist/admin/utils/api.mjs +6 -0
  116. package/dist/admin/utils/api.mjs.map +1 -0
  117. package/dist/admin/utils/prefixPluginTranslations.js +11 -0
  118. package/dist/admin/utils/prefixPluginTranslations.js.map +1 -0
  119. package/dist/admin/utils/prefixPluginTranslations.mjs +9 -0
  120. package/dist/admin/utils/prefixPluginTranslations.mjs.map +1 -0
  121. package/dist/admin/utils/time.js +42 -0
  122. package/dist/admin/utils/time.js.map +1 -0
  123. package/dist/admin/utils/time.mjs +39 -0
  124. package/dist/admin/utils/time.mjs.map +1 -0
  125. package/dist/admin/validation/schemas.js +65 -0
  126. package/dist/admin/validation/schemas.js.map +1 -0
  127. package/dist/admin/validation/schemas.mjs +43 -0
  128. package/dist/admin/validation/schemas.mjs.map +1 -0
  129. package/dist/server/bootstrap.js +68 -0
  130. package/dist/server/bootstrap.js.map +1 -0
  131. package/dist/server/bootstrap.mjs +66 -0
  132. package/dist/server/bootstrap.mjs.map +1 -0
  133. package/dist/server/constants.js +74 -0
  134. package/dist/server/constants.js.map +1 -0
  135. package/dist/server/constants.mjs +69 -0
  136. package/dist/server/constants.mjs.map +1 -0
  137. package/dist/server/content-types/index.js +12 -0
  138. package/dist/server/content-types/index.js.map +1 -0
  139. package/dist/server/content-types/index.mjs +10 -0
  140. package/dist/server/content-types/index.mjs.map +1 -0
  141. package/dist/server/content-types/release/index.js +10 -0
  142. package/dist/server/content-types/release/index.js.map +1 -0
  143. package/dist/server/content-types/release/index.mjs +8 -0
  144. package/dist/server/content-types/release/index.mjs.map +1 -0
  145. package/dist/server/content-types/release/schema.js +58 -0
  146. package/dist/server/content-types/release/schema.js.map +1 -0
  147. package/dist/server/content-types/release/schema.mjs +56 -0
  148. package/dist/server/content-types/release/schema.mjs.map +1 -0
  149. package/dist/server/content-types/release-action/index.js +10 -0
  150. package/dist/server/content-types/release-action/index.js.map +1 -0
  151. package/dist/server/content-types/release-action/index.mjs +8 -0
  152. package/dist/server/content-types/release-action/index.mjs.map +1 -0
  153. package/dist/server/content-types/release-action/schema.js +55 -0
  154. package/dist/server/content-types/release-action/schema.js.map +1 -0
  155. package/dist/server/content-types/release-action/schema.mjs +53 -0
  156. package/dist/server/content-types/release-action/schema.mjs.map +1 -0
  157. package/dist/server/controllers/index.js +14 -0
  158. package/dist/server/controllers/index.js.map +1 -0
  159. package/dist/server/controllers/index.mjs +12 -0
  160. package/dist/server/controllers/index.mjs.map +1 -0
  161. package/dist/server/controllers/release-action.js +150 -0
  162. package/dist/server/controllers/release-action.js.map +1 -0
  163. package/dist/server/controllers/release-action.mjs +148 -0
  164. package/dist/server/controllers/release-action.mjs.map +1 -0
  165. package/dist/server/controllers/release.js +302 -0
  166. package/dist/server/controllers/release.js.map +1 -0
  167. package/dist/server/controllers/release.mjs +300 -0
  168. package/dist/server/controllers/release.mjs.map +1 -0
  169. package/dist/server/controllers/settings.js +37 -0
  170. package/dist/server/controllers/settings.js.map +1 -0
  171. package/dist/server/controllers/settings.mjs +35 -0
  172. package/dist/server/controllers/settings.mjs.map +1 -0
  173. package/dist/server/controllers/validation/release-action.js +34 -0
  174. package/dist/server/controllers/validation/release-action.js.map +1 -0
  175. package/dist/server/controllers/validation/release-action.mjs +30 -0
  176. package/dist/server/controllers/validation/release-action.mjs.map +1 -0
  177. package/dist/server/controllers/validation/release.js +26 -0
  178. package/dist/server/controllers/validation/release.js.map +1 -0
  179. package/dist/server/controllers/validation/release.mjs +22 -0
  180. package/dist/server/controllers/validation/release.mjs.map +1 -0
  181. package/dist/server/controllers/validation/settings.js +32 -0
  182. package/dist/server/controllers/validation/settings.js.map +1 -0
  183. package/dist/server/controllers/validation/settings.mjs +10 -0
  184. package/dist/server/controllers/validation/settings.mjs.map +1 -0
  185. package/dist/server/destroy.js +15 -0
  186. package/dist/server/destroy.js.map +1 -0
  187. package/dist/server/destroy.mjs +13 -0
  188. package/dist/server/destroy.mjs.map +1 -0
  189. package/dist/server/index.js +25 -1574
  190. package/dist/server/index.js.map +1 -1
  191. package/dist/server/index.mjs +24 -1554
  192. package/dist/server/index.mjs.map +1 -1
  193. package/dist/server/middlewares/documents.js +104 -0
  194. package/dist/server/middlewares/documents.js.map +1 -0
  195. package/dist/server/middlewares/documents.mjs +101 -0
  196. package/dist/server/middlewares/documents.mjs.map +1 -0
  197. package/dist/server/migrations/database/5.0.0-document-id-in-actions.js +51 -0
  198. package/dist/server/migrations/database/5.0.0-document-id-in-actions.js.map +1 -0
  199. package/dist/server/migrations/database/5.0.0-document-id-in-actions.mjs +49 -0
  200. package/dist/server/migrations/database/5.0.0-document-id-in-actions.mjs.map +1 -0
  201. package/dist/server/migrations/index.js +205 -0
  202. package/dist/server/migrations/index.js.map +1 -0
  203. package/dist/server/migrations/index.mjs +198 -0
  204. package/dist/server/migrations/index.mjs.map +1 -0
  205. package/dist/server/register.js +23 -0
  206. package/dist/server/register.js.map +1 -0
  207. package/dist/server/register.mjs +21 -0
  208. package/dist/server/register.mjs.map +1 -0
  209. package/dist/server/routes/index.js +14 -0
  210. package/dist/server/routes/index.js.map +1 -0
  211. package/dist/server/routes/index.mjs +12 -0
  212. package/dist/server/routes/index.mjs.map +1 -0
  213. package/dist/server/routes/release-action.js +100 -0
  214. package/dist/server/routes/release-action.js.map +1 -0
  215. package/dist/server/routes/release-action.mjs +98 -0
  216. package/dist/server/routes/release-action.mjs.map +1 -0
  217. package/dist/server/routes/release.js +154 -0
  218. package/dist/server/routes/release.js.map +1 -0
  219. package/dist/server/routes/release.mjs +152 -0
  220. package/dist/server/routes/release.mjs.map +1 -0
  221. package/dist/server/routes/settings.js +46 -0
  222. package/dist/server/routes/settings.js.map +1 -0
  223. package/dist/server/routes/settings.mjs +44 -0
  224. package/dist/server/routes/settings.mjs.map +1 -0
  225. package/dist/server/services/index.js +18 -0
  226. package/dist/server/services/index.js.map +1 -0
  227. package/dist/server/services/index.mjs +16 -0
  228. package/dist/server/services/index.mjs.map +1 -0
  229. package/dist/server/services/release-action.js +324 -0
  230. package/dist/server/services/release-action.js.map +1 -0
  231. package/dist/server/services/release-action.mjs +322 -0
  232. package/dist/server/services/release-action.mjs.map +1 -0
  233. package/dist/server/services/release.js +324 -0
  234. package/dist/server/services/release.js.map +1 -0
  235. package/dist/server/services/release.mjs +322 -0
  236. package/dist/server/services/release.mjs.map +1 -0
  237. package/dist/server/services/scheduling.js +70 -0
  238. package/dist/server/services/scheduling.js.map +1 -0
  239. package/dist/server/services/scheduling.mjs +68 -0
  240. package/dist/server/services/scheduling.mjs.map +1 -0
  241. package/dist/server/services/settings.js +34 -0
  242. package/dist/server/services/settings.js.map +1 -0
  243. package/dist/server/services/settings.mjs +32 -0
  244. package/dist/server/services/settings.mjs.map +1 -0
  245. package/dist/server/services/validation.js +91 -0
  246. package/dist/server/services/validation.js.map +1 -0
  247. package/dist/server/services/validation.mjs +86 -0
  248. package/dist/server/services/validation.mjs.map +1 -0
  249. package/dist/server/src/bootstrap.d.ts +2 -2
  250. package/dist/server/src/bootstrap.d.ts.map +1 -1
  251. package/dist/server/src/constants.d.ts +11 -2
  252. package/dist/server/src/constants.d.ts.map +1 -1
  253. package/dist/server/src/content-types/index.d.ts +3 -5
  254. package/dist/server/src/content-types/index.d.ts.map +1 -1
  255. package/dist/server/src/content-types/release-action/index.d.ts +3 -5
  256. package/dist/server/src/content-types/release-action/index.d.ts.map +1 -1
  257. package/dist/server/src/content-types/release-action/schema.d.ts +3 -5
  258. package/dist/server/src/content-types/release-action/schema.d.ts.map +1 -1
  259. package/dist/server/src/controllers/index.d.ts +8 -1
  260. package/dist/server/src/controllers/index.d.ts.map +1 -1
  261. package/dist/server/src/controllers/release-action.d.ts +1 -0
  262. package/dist/server/src/controllers/release-action.d.ts.map +1 -1
  263. package/dist/server/src/controllers/release.d.ts +8 -1
  264. package/dist/server/src/controllers/release.d.ts.map +1 -1
  265. package/dist/server/src/controllers/settings.d.ts +11 -0
  266. package/dist/server/src/controllers/settings.d.ts.map +1 -0
  267. package/dist/server/src/controllers/validation/release-action.d.ts +13 -2
  268. package/dist/server/src/controllers/validation/release-action.d.ts.map +1 -1
  269. package/dist/server/src/controllers/validation/release.d.ts +2 -0
  270. package/dist/server/src/controllers/validation/release.d.ts.map +1 -1
  271. package/dist/server/src/controllers/validation/settings.d.ts +3 -0
  272. package/dist/server/src/controllers/validation/settings.d.ts.map +1 -0
  273. package/dist/server/src/destroy.d.ts +2 -2
  274. package/dist/server/src/destroy.d.ts.map +1 -1
  275. package/dist/server/src/index.d.ts +1465 -1446
  276. package/dist/server/src/index.d.ts.map +1 -1
  277. package/dist/server/src/middlewares/documents.d.ts +6 -0
  278. package/dist/server/src/middlewares/documents.d.ts.map +1 -0
  279. package/dist/server/src/migrations/database/5.0.0-document-id-in-actions.d.ts +9 -0
  280. package/dist/server/src/migrations/database/5.0.0-document-id-in-actions.d.ts.map +1 -0
  281. package/dist/server/src/migrations/index.d.ts +3 -0
  282. package/dist/server/src/migrations/index.d.ts.map +1 -1
  283. package/dist/server/src/register.d.ts +2 -2
  284. package/dist/server/src/register.d.ts.map +1 -1
  285. package/dist/server/src/routes/index.d.ts +16 -0
  286. package/dist/server/src/routes/index.d.ts.map +1 -1
  287. package/dist/server/src/routes/release-action.d.ts.map +1 -1
  288. package/dist/server/src/routes/release.d.ts.map +1 -1
  289. package/dist/server/src/routes/settings.d.ts +18 -0
  290. package/dist/server/src/routes/settings.d.ts.map +1 -0
  291. package/dist/server/src/services/index.d.ts +1429 -1431
  292. package/dist/server/src/services/index.d.ts.map +1 -1
  293. package/dist/server/src/services/release-action.d.ts +34 -0
  294. package/dist/server/src/services/release-action.d.ts.map +1 -0
  295. package/dist/server/src/services/release.d.ts +8 -43
  296. package/dist/server/src/services/release.d.ts.map +1 -1
  297. package/dist/server/src/services/scheduling.d.ts +6 -6
  298. package/dist/server/src/services/scheduling.d.ts.map +1 -1
  299. package/dist/server/src/services/settings.d.ts +13 -0
  300. package/dist/server/src/services/settings.d.ts.map +1 -0
  301. package/dist/server/src/services/validation.d.ts +7 -3
  302. package/dist/server/src/services/validation.d.ts.map +1 -1
  303. package/dist/server/src/utils/index.d.ts +33 -12
  304. package/dist/server/src/utils/index.d.ts.map +1 -1
  305. package/dist/server/utils/index.js +94 -0
  306. package/dist/server/utils/index.js.map +1 -0
  307. package/dist/server/utils/index.mjs +88 -0
  308. package/dist/server/utils/index.mjs.map +1 -0
  309. package/dist/shared/contracts/release-actions.d.ts +44 -13
  310. package/dist/shared/contracts/release-actions.d.ts.map +1 -1
  311. package/dist/shared/contracts/releases.d.ts +24 -7
  312. package/dist/shared/contracts/releases.d.ts.map +1 -1
  313. package/dist/shared/contracts/settings.d.ts +38 -0
  314. package/dist/shared/contracts/settings.d.ts.map +1 -0
  315. package/dist/shared/types.d.ts +2 -3
  316. package/dist/shared/types.d.ts.map +1 -1
  317. package/package.json +36 -31
  318. package/dist/_chunks/App-PQlYzNfb.mjs +0 -1337
  319. package/dist/_chunks/App-PQlYzNfb.mjs.map +0 -1
  320. package/dist/_chunks/App-lzeJz92X.js +0 -1360
  321. package/dist/_chunks/App-lzeJz92X.js.map +0 -1
  322. package/dist/_chunks/PurchaseContentReleases-Clm0iACO.mjs +0 -51
  323. package/dist/_chunks/PurchaseContentReleases-Clm0iACO.mjs.map +0 -1
  324. package/dist/_chunks/PurchaseContentReleases-YhAPgpG9.js +0 -51
  325. package/dist/_chunks/PurchaseContentReleases-YhAPgpG9.js.map +0 -1
  326. package/dist/_chunks/en-WuuhP6Bn.mjs +0 -78
  327. package/dist/_chunks/en-WuuhP6Bn.mjs.map +0 -1
  328. package/dist/_chunks/en-gcJJ5htG.js +0 -78
  329. package/dist/_chunks/en-gcJJ5htG.js.map +0 -1
  330. package/dist/_chunks/index--4AgLDzb.mjs +0 -1007
  331. package/dist/_chunks/index--4AgLDzb.mjs.map +0 -1
  332. package/dist/_chunks/index-Nf1JfI-m.js +0 -1028
  333. package/dist/_chunks/index-Nf1JfI-m.js.map +0 -1
  334. package/dist/admin/src/components/CMReleasesContainer.d.ts +0 -1
  335. package/dist/admin/src/services/axios.d.ts +0 -29
  336. package/dist/shared/validation-schemas.d.ts +0 -2
  337. package/dist/shared/validation-schemas.d.ts.map +0 -1
  338. package/strapi-server.js +0 -3
@@ -1,1337 +0,0 @@
1
- import { jsx, jsxs, Fragment } from "react/jsx-runtime";
2
- import { useLicenseLimits, Page, Pagination, BackButton, Table, unstable_useDocument } from "@strapi/admin/strapi-admin";
3
- import { useLocation, useNavigate, useParams, Navigate, Link as Link$1, Routes, Route } from "react-router-dom";
4
- import { g as getTimezoneOffset, p as pluginId, u as useGetReleasesQuery, a as useCreateReleaseMutation, P as PERMISSIONS, i as isAxiosError, b as useGetReleaseQuery, c as useUpdateReleaseMutation, d as useDeleteReleaseMutation, e as usePublishReleaseMutation, f as useTypedDispatch, h as useGetReleaseActionsQuery, j as useUpdateReleaseActionMutation, R as ReleaseActionOptions, k as ReleaseActionMenu, r as releaseApi } from "./index--4AgLDzb.mjs";
5
- import * as React from "react";
6
- import { ModalLayout, ModalHeader, Typography, ModalBody, Flex, TextInput, Box, Checkbox, DatePicker, TimePicker, ModalFooter, Button, Combobox, ComboboxOption, Alert, Main, HeaderLayout, ContentLayout, TabGroup, Tabs, Tab, Divider, TabPanels, TabPanel, EmptyStateLayout, Grid, GridItem, Badge, IconButton, SingleSelect, SingleSelectOption, Tr, Td, Icon, Tooltip } from "@strapi/design-system";
7
- import { Link, Menu, LinkButton } from "@strapi/design-system/v2";
8
- import { useNotification, useAPIErrorHandler, useQueryParams, useTracking, CheckPermissions, ConfirmDialog, useRBAC } from "@strapi/helper-plugin";
9
- import { Plus, EmptyDocuments, Pencil, Trash, More, CrossCircle, CheckCircle } from "@strapi/icons";
10
- import format from "date-fns/format";
11
- import { zonedTimeToUtc, utcToZonedTime } from "date-fns-tz";
12
- import { useIntl } from "react-intl";
13
- import styled from "styled-components";
14
- import { intervalToDuration, isPast, formatISO, parse } from "date-fns";
15
- import { Formik, Form, useFormikContext } from "formik";
16
- import * as yup from "yup";
17
- import "@reduxjs/toolkit/query";
18
- import "axios";
19
- import "@reduxjs/toolkit/query/react";
20
- import "react-redux";
21
- const intervals = ["years", "months", "days", "hours", "minutes", "seconds"];
22
- const RelativeTime = React.forwardRef(
23
- ({ timestamp, customIntervals = [], ...restProps }, forwardedRef) => {
24
- const { formatRelativeTime, formatDate, formatTime } = useIntl();
25
- const interval = intervalToDuration({
26
- start: timestamp,
27
- end: Date.now()
28
- // see https://github.com/date-fns/date-fns/issues/2891 – No idea why it's all partial it returns it every time.
29
- });
30
- const unit = intervals.find((intervalUnit) => {
31
- return interval[intervalUnit] > 0 && Object.keys(interval).includes(intervalUnit);
32
- });
33
- const relativeTime = isPast(timestamp) ? -interval[unit] : interval[unit];
34
- const customInterval = customIntervals.find(
35
- (custom) => interval[custom.unit] < custom.threshold
36
- );
37
- const displayText = customInterval ? customInterval.text : formatRelativeTime(relativeTime, unit, { numeric: "auto" });
38
- return /* @__PURE__ */ jsx(
39
- "time",
40
- {
41
- ref: forwardedRef,
42
- dateTime: timestamp.toISOString(),
43
- role: "time",
44
- title: `${formatDate(timestamp)} ${formatTime(timestamp)}`,
45
- ...restProps,
46
- children: displayText
47
- }
48
- );
49
- }
50
- );
51
- const RELEASE_SCHEMA = yup.object().shape({
52
- name: yup.string().trim().required(),
53
- scheduledAt: yup.string().nullable(),
54
- isScheduled: yup.boolean().optional(),
55
- time: yup.string().when("isScheduled", {
56
- is: true,
57
- then: yup.string().trim().required(),
58
- otherwise: yup.string().nullable()
59
- }),
60
- timezone: yup.string().when("isScheduled", {
61
- is: true,
62
- then: yup.string().required().nullable(),
63
- otherwise: yup.string().nullable()
64
- }),
65
- date: yup.string().when("isScheduled", {
66
- is: true,
67
- then: yup.string().required().nullable(),
68
- otherwise: yup.string().nullable()
69
- })
70
- }).required().noUnknown();
71
- const ReleaseModal = ({
72
- handleClose,
73
- handleSubmit,
74
- initialValues,
75
- isLoading = false
76
- }) => {
77
- const { formatMessage } = useIntl();
78
- const { pathname } = useLocation();
79
- const isCreatingRelease = pathname === `/plugins/${pluginId}`;
80
- const IsSchedulingEnabled = window.strapi.future.isEnabled("contentReleasesScheduling");
81
- const { timezoneList, systemTimezone = { value: "UTC+00:00-Africa/Abidjan " } } = getTimezones(
82
- initialValues.scheduledAt ? new Date(initialValues.scheduledAt) : /* @__PURE__ */ new Date()
83
- );
84
- const getScheduledTimestamp = (values) => {
85
- const { date, time, timezone } = values;
86
- if (!date || !time || !timezone)
87
- return null;
88
- const formattedDate = parse(time, "HH:mm", new Date(date));
89
- const timezoneWithoutOffset = timezone.split("_")[1];
90
- return zonedTimeToUtc(formattedDate, timezoneWithoutOffset);
91
- };
92
- const getTimezoneWithOffset = () => {
93
- const currentTimezone = timezoneList.find(
94
- (timezone) => timezone.value.split("_")[1] === initialValues.timezone
95
- );
96
- return currentTimezone?.value || systemTimezone.value;
97
- };
98
- return /* @__PURE__ */ jsxs(ModalLayout, { onClose: handleClose, labelledBy: "title", children: [
99
- /* @__PURE__ */ jsx(ModalHeader, { children: /* @__PURE__ */ jsx(Typography, { id: "title", fontWeight: "bold", textColor: "neutral800", children: formatMessage(
100
- {
101
- id: "content-releases.modal.title",
102
- defaultMessage: "{isCreatingRelease, select, true {New release} other {Edit release}}"
103
- },
104
- { isCreatingRelease }
105
- ) }) }),
106
- /* @__PURE__ */ jsx(
107
- Formik,
108
- {
109
- onSubmit: (values) => {
110
- handleSubmit({
111
- ...values,
112
- timezone: values.timezone ? values.timezone.split("_")[1] : null,
113
- scheduledAt: values.isScheduled ? getScheduledTimestamp(values) : null
114
- });
115
- },
116
- initialValues: {
117
- ...initialValues,
118
- timezone: initialValues.timezone ? getTimezoneWithOffset() : systemTimezone.value
119
- },
120
- validationSchema: RELEASE_SCHEMA,
121
- validateOnChange: false,
122
- children: ({ values, errors, handleChange, setFieldValue }) => /* @__PURE__ */ jsxs(Form, { children: [
123
- /* @__PURE__ */ jsx(ModalBody, { children: /* @__PURE__ */ jsxs(Flex, { direction: "column", alignItems: "stretch", gap: 6, children: [
124
- /* @__PURE__ */ jsx(
125
- TextInput,
126
- {
127
- label: formatMessage({
128
- id: "content-releases.modal.form.input.label.release-name",
129
- defaultMessage: "Name"
130
- }),
131
- name: "name",
132
- value: values.name,
133
- error: errors.name,
134
- onChange: handleChange,
135
- required: true
136
- }
137
- ),
138
- IsSchedulingEnabled && /* @__PURE__ */ jsxs(Fragment, { children: [
139
- /* @__PURE__ */ jsx(Box, { width: "max-content", children: /* @__PURE__ */ jsx(
140
- Checkbox,
141
- {
142
- name: "isScheduled",
143
- value: values.isScheduled,
144
- onChange: (event) => {
145
- setFieldValue("isScheduled", event.target.checked);
146
- if (!event.target.checked) {
147
- setFieldValue("date", null);
148
- setFieldValue("time", "");
149
- setFieldValue("timezone", null);
150
- } else {
151
- setFieldValue("date", initialValues.date);
152
- setFieldValue("time", initialValues.time);
153
- setFieldValue(
154
- "timezone",
155
- initialValues.timezone ?? systemTimezone?.value
156
- );
157
- }
158
- },
159
- children: /* @__PURE__ */ jsx(
160
- Typography,
161
- {
162
- textColor: values.isScheduled ? "primary600" : "neutral800",
163
- fontWeight: values.isScheduled ? "semiBold" : "regular",
164
- children: formatMessage({
165
- id: "modal.form.input.label.schedule-release",
166
- defaultMessage: "Schedule release"
167
- })
168
- }
169
- )
170
- }
171
- ) }),
172
- values.isScheduled && /* @__PURE__ */ jsxs(Fragment, { children: [
173
- /* @__PURE__ */ jsxs(Flex, { gap: 4, alignItems: "start", children: [
174
- /* @__PURE__ */ jsx(Box, { width: "100%", children: /* @__PURE__ */ jsx(
175
- DatePicker,
176
- {
177
- label: formatMessage({
178
- id: "content-releases.modal.form.input.label.date",
179
- defaultMessage: "Date"
180
- }),
181
- name: "date",
182
- error: errors.date,
183
- onChange: (date) => {
184
- const isoFormatDate = date ? formatISO(date, { representation: "date" }) : null;
185
- setFieldValue("date", isoFormatDate);
186
- },
187
- clearLabel: formatMessage({
188
- id: "content-releases.modal.form.input.clearLabel",
189
- defaultMessage: "Clear"
190
- }),
191
- onClear: () => {
192
- setFieldValue("date", null);
193
- },
194
- selectedDate: values.date || void 0,
195
- required: true
196
- }
197
- ) }),
198
- /* @__PURE__ */ jsx(Box, { width: "100%", children: /* @__PURE__ */ jsx(
199
- TimePicker,
200
- {
201
- label: formatMessage({
202
- id: "content-releases.modal.form.input.label.time",
203
- defaultMessage: "Time"
204
- }),
205
- name: "time",
206
- error: errors.time,
207
- onChange: (time) => {
208
- setFieldValue("time", time);
209
- },
210
- clearLabel: formatMessage({
211
- id: "content-releases.modal.form.input.clearLabel",
212
- defaultMessage: "Clear"
213
- }),
214
- onClear: () => {
215
- setFieldValue("time", "");
216
- },
217
- value: values.time || void 0,
218
- required: true
219
- }
220
- ) })
221
- ] }),
222
- /* @__PURE__ */ jsx(TimezoneComponent, { timezoneOptions: timezoneList })
223
- ] })
224
- ] })
225
- ] }) }),
226
- /* @__PURE__ */ jsx(
227
- ModalFooter,
228
- {
229
- startActions: /* @__PURE__ */ jsx(Button, { onClick: handleClose, variant: "tertiary", name: "cancel", children: formatMessage({ id: "cancel", defaultMessage: "Cancel" }) }),
230
- endActions: /* @__PURE__ */ jsx(Button, { name: "submit", loading: isLoading, type: "submit", children: formatMessage(
231
- {
232
- id: "content-releases.modal.form.button.submit",
233
- defaultMessage: "{isCreatingRelease, select, true {Continue} other {Save}}"
234
- },
235
- { isCreatingRelease }
236
- ) })
237
- }
238
- )
239
- ] })
240
- }
241
- )
242
- ] });
243
- };
244
- const getTimezones = (selectedDate) => {
245
- const timezoneList = Intl.supportedValuesOf("timeZone").map((timezone) => {
246
- const utcOffset = getTimezoneOffset(timezone, selectedDate);
247
- return { offset: utcOffset, value: `${utcOffset}_${timezone}` };
248
- });
249
- const systemTimezone = timezoneList.find(
250
- (timezone) => timezone.value.split("_")[1] === Intl.DateTimeFormat().resolvedOptions().timeZone
251
- );
252
- return { timezoneList, systemTimezone };
253
- };
254
- const TimezoneComponent = ({ timezoneOptions }) => {
255
- const { values, errors, setFieldValue } = useFormikContext();
256
- const { formatMessage } = useIntl();
257
- const [timezoneList, setTimezoneList] = React.useState(timezoneOptions);
258
- React.useEffect(() => {
259
- if (values.date) {
260
- const { timezoneList: timezoneList2 } = getTimezones(new Date(values.date));
261
- setTimezoneList(timezoneList2);
262
- const updatedTimezone = values.timezone && timezoneList2.find((tz) => tz.value.split("_")[1] === values.timezone.split("_")[1]);
263
- if (updatedTimezone) {
264
- setFieldValue("timezone", updatedTimezone.value);
265
- }
266
- }
267
- }, [setFieldValue, values.date, values.timezone]);
268
- return /* @__PURE__ */ jsx(
269
- Combobox,
270
- {
271
- label: formatMessage({
272
- id: "content-releases.modal.form.input.label.timezone",
273
- defaultMessage: "Timezone"
274
- }),
275
- name: "timezone",
276
- value: values.timezone || void 0,
277
- textValue: values.timezone ? values.timezone.replace("_", " ") : void 0,
278
- onChange: (timezone) => {
279
- setFieldValue("timezone", timezone);
280
- },
281
- onTextValueChange: (timezone) => {
282
- setFieldValue("timezone", timezone);
283
- },
284
- onClear: () => {
285
- setFieldValue("timezone", "");
286
- },
287
- error: errors.timezone,
288
- required: true,
289
- children: timezoneList.map((timezone) => /* @__PURE__ */ jsx(ComboboxOption, { value: timezone.value, children: timezone.value.replace("_", " ") }, timezone.value))
290
- }
291
- );
292
- };
293
- const LinkCard = styled(Link)`
294
- display: block;
295
- `;
296
- const CapitalizeRelativeTime = styled(RelativeTime)`
297
- text-transform: capitalize;
298
- `;
299
- const getBadgeProps = (status) => {
300
- let color;
301
- switch (status) {
302
- case "ready":
303
- color = "success";
304
- break;
305
- case "blocked":
306
- color = "warning";
307
- break;
308
- case "failed":
309
- color = "danger";
310
- break;
311
- case "done":
312
- color = "primary";
313
- break;
314
- case "empty":
315
- default:
316
- color = "neutral";
317
- }
318
- return {
319
- textColor: `${color}600`,
320
- backgroundColor: `${color}100`,
321
- borderColor: `${color}200`
322
- };
323
- };
324
- const ReleasesGrid = ({ sectionTitle, releases = [], isError = false }) => {
325
- const { formatMessage } = useIntl();
326
- const IsSchedulingEnabled = window.strapi.future.isEnabled("contentReleasesScheduling");
327
- if (isError) {
328
- return /* @__PURE__ */ jsx(Page.Error, {});
329
- }
330
- if (releases?.length === 0) {
331
- return /* @__PURE__ */ jsx(
332
- EmptyStateLayout,
333
- {
334
- content: formatMessage(
335
- {
336
- id: "content-releases.page.Releases.tab.emptyEntries",
337
- defaultMessage: "No releases"
338
- },
339
- {
340
- target: sectionTitle
341
- }
342
- ),
343
- icon: /* @__PURE__ */ jsx(EmptyDocuments, { width: "10rem" })
344
- }
345
- );
346
- }
347
- return /* @__PURE__ */ jsx(Grid, { gap: 4, children: releases.map(({ id, name, actions, scheduledAt, status }) => /* @__PURE__ */ jsx(GridItem, { col: 3, s: 6, xs: 12, children: /* @__PURE__ */ jsx(LinkCard, { href: `content-releases/${id}`, isExternal: false, children: /* @__PURE__ */ jsxs(
348
- Flex,
349
- {
350
- direction: "column",
351
- justifyContent: "space-between",
352
- padding: 4,
353
- hasRadius: true,
354
- background: "neutral0",
355
- shadow: "tableShadow",
356
- height: "100%",
357
- width: "100%",
358
- alignItems: "start",
359
- gap: 4,
360
- children: [
361
- /* @__PURE__ */ jsxs(Flex, { direction: "column", alignItems: "start", gap: 1, children: [
362
- /* @__PURE__ */ jsx(Typography, { as: "h3", variant: "delta", fontWeight: "bold", children: name }),
363
- /* @__PURE__ */ jsx(Typography, { variant: "pi", textColor: "neutral600", children: IsSchedulingEnabled ? scheduledAt ? /* @__PURE__ */ jsx(CapitalizeRelativeTime, { timestamp: new Date(scheduledAt) }) : formatMessage({
364
- id: "content-releases.pages.Releases.not-scheduled",
365
- defaultMessage: "Not scheduled"
366
- }) : formatMessage(
367
- {
368
- id: "content-releases.page.Releases.release-item.entries",
369
- defaultMessage: "{number, plural, =0 {No entries} one {# entry} other {# entries}}"
370
- },
371
- { number: actions.meta.count }
372
- ) })
373
- ] }),
374
- /* @__PURE__ */ jsx(Badge, { ...getBadgeProps(status), children: status })
375
- ]
376
- }
377
- ) }) }, id)) });
378
- };
379
- const StyledAlert = styled(Alert)`
380
- button {
381
- display: none;
382
- }
383
- p + div {
384
- margin-left: auto;
385
- }
386
- `;
387
- const INITIAL_FORM_VALUES = {
388
- name: "",
389
- date: null,
390
- time: "",
391
- // Remove future flag check after Scheduling Beta release and replace with true as creating new release should include scheduling by default
392
- isScheduled: window.strapi.future.isEnabled("contentReleasesScheduling"),
393
- scheduledAt: null,
394
- timezone: null
395
- };
396
- const ReleasesPage = () => {
397
- const tabRef = React.useRef(null);
398
- const location = useLocation();
399
- const [releaseModalShown, setReleaseModalShown] = React.useState(false);
400
- const toggleNotification = useNotification();
401
- const { formatMessage } = useIntl();
402
- const navigate = useNavigate();
403
- const { formatAPIError } = useAPIErrorHandler();
404
- const [{ query }, setQuery] = useQueryParams();
405
- const response = useGetReleasesQuery(query);
406
- const [createRelease, { isLoading: isSubmittingForm }] = useCreateReleaseMutation();
407
- const { getFeature } = useLicenseLimits();
408
- const { maximumReleases = 3 } = getFeature("cms-content-releases");
409
- const { trackUsage } = useTracking();
410
- const { isLoading, isSuccess, isError } = response;
411
- const activeTab = response?.currentData?.meta?.activeTab || "pending";
412
- const activeTabIndex = ["pending", "done"].indexOf(activeTab);
413
- React.useEffect(() => {
414
- if (location?.state?.errors) {
415
- toggleNotification({
416
- type: "warning",
417
- title: formatMessage({
418
- id: "content-releases.pages.Releases.notification.error.title",
419
- defaultMessage: "Your request could not be processed."
420
- }),
421
- message: formatMessage({
422
- id: "content-releases.pages.Releases.notification.error.message",
423
- defaultMessage: "Please try again or open another release."
424
- })
425
- });
426
- navigate("", { replace: true, state: null });
427
- }
428
- }, [formatMessage, location?.state?.errors, navigate, toggleNotification]);
429
- React.useEffect(() => {
430
- if (tabRef.current) {
431
- tabRef.current._handlers.setSelectedTabIndex(activeTabIndex);
432
- }
433
- }, [activeTabIndex]);
434
- const toggleAddReleaseModal = () => {
435
- setReleaseModalShown((prev) => !prev);
436
- };
437
- if (isLoading) {
438
- return /* @__PURE__ */ jsx(Page.Loading, {});
439
- }
440
- const totalPendingReleases = isSuccess && response.currentData?.meta?.pendingReleasesCount || 0;
441
- const hasReachedMaximumPendingReleases = totalPendingReleases >= maximumReleases;
442
- const handleTabChange = (index) => {
443
- setQuery({
444
- ...query,
445
- page: 1,
446
- pageSize: response?.currentData?.meta?.pagination?.pageSize || 16,
447
- filters: {
448
- releasedAt: {
449
- $notNull: index === 0 ? false : true
450
- }
451
- }
452
- });
453
- };
454
- const handleAddRelease = async ({ name, scheduledAt, timezone }) => {
455
- const response2 = await createRelease({
456
- name,
457
- scheduledAt,
458
- timezone
459
- });
460
- if ("data" in response2) {
461
- toggleNotification({
462
- type: "success",
463
- message: formatMessage({
464
- id: "content-releases.modal.release-created-notification-success",
465
- defaultMessage: "Release created."
466
- })
467
- });
468
- trackUsage("didCreateRelease");
469
- navigate(response2.data.data.id.toString());
470
- } else if (isAxiosError(response2.error)) {
471
- toggleNotification({
472
- type: "warning",
473
- message: formatAPIError(response2.error)
474
- });
475
- } else {
476
- toggleNotification({
477
- type: "warning",
478
- message: formatMessage({ id: "notification.error", defaultMessage: "An error occurred" })
479
- });
480
- }
481
- };
482
- return /* @__PURE__ */ jsxs(Main, { "aria-busy": isLoading, children: [
483
- /* @__PURE__ */ jsx(
484
- HeaderLayout,
485
- {
486
- title: formatMessage({
487
- id: "content-releases.pages.Releases.title",
488
- defaultMessage: "Releases"
489
- }),
490
- subtitle: formatMessage({
491
- id: "content-releases.pages.Releases.header-subtitle",
492
- defaultMessage: "Create and manage content updates"
493
- }),
494
- primaryAction: /* @__PURE__ */ jsx(CheckPermissions, { permissions: PERMISSIONS.create, children: /* @__PURE__ */ jsx(
495
- Button,
496
- {
497
- startIcon: /* @__PURE__ */ jsx(Plus, {}),
498
- onClick: toggleAddReleaseModal,
499
- disabled: hasReachedMaximumPendingReleases,
500
- children: formatMessage({
501
- id: "content-releases.header.actions.add-release",
502
- defaultMessage: "New release"
503
- })
504
- }
505
- ) })
506
- }
507
- ),
508
- /* @__PURE__ */ jsx(ContentLayout, { children: /* @__PURE__ */ jsxs(Fragment, { children: [
509
- hasReachedMaximumPendingReleases && /* @__PURE__ */ jsx(
510
- StyledAlert,
511
- {
512
- marginBottom: 6,
513
- action: /* @__PURE__ */ jsx(Link, { href: "https://strapi.io/pricing-cloud", isExternal: true, children: formatMessage({
514
- id: "content-releases.pages.Releases.max-limit-reached.action",
515
- defaultMessage: "Explore plans"
516
- }) }),
517
- title: formatMessage(
518
- {
519
- id: "content-releases.pages.Releases.max-limit-reached.title",
520
- defaultMessage: "You have reached the {number} pending {number, plural, one {release} other {releases}} limit."
521
- },
522
- { number: maximumReleases }
523
- ),
524
- onClose: () => {
525
- },
526
- closeLabel: "",
527
- children: formatMessage({
528
- id: "content-releases.pages.Releases.max-limit-reached.message",
529
- defaultMessage: "Upgrade to manage an unlimited number of releases."
530
- })
531
- }
532
- ),
533
- /* @__PURE__ */ jsxs(
534
- TabGroup,
535
- {
536
- label: formatMessage({
537
- id: "content-releases.pages.Releases.tab-group.label",
538
- defaultMessage: "Releases list"
539
- }),
540
- variant: "simple",
541
- initialSelectedTabIndex: activeTabIndex,
542
- onTabChange: handleTabChange,
543
- ref: tabRef,
544
- children: [
545
- /* @__PURE__ */ jsxs(Box, { paddingBottom: 8, children: [
546
- /* @__PURE__ */ jsxs(Tabs, { children: [
547
- /* @__PURE__ */ jsx(Tab, { children: formatMessage(
548
- {
549
- id: "content-releases.pages.Releases.tab.pending",
550
- defaultMessage: "Pending ({count})"
551
- },
552
- {
553
- count: totalPendingReleases
554
- }
555
- ) }),
556
- /* @__PURE__ */ jsx(Tab, { children: formatMessage({
557
- id: "content-releases.pages.Releases.tab.done",
558
- defaultMessage: "Done"
559
- }) })
560
- ] }),
561
- /* @__PURE__ */ jsx(Divider, {})
562
- ] }),
563
- /* @__PURE__ */ jsxs(TabPanels, { children: [
564
- /* @__PURE__ */ jsx(TabPanel, { children: /* @__PURE__ */ jsx(
565
- ReleasesGrid,
566
- {
567
- sectionTitle: "pending",
568
- releases: response?.currentData?.data,
569
- isError
570
- }
571
- ) }),
572
- /* @__PURE__ */ jsx(TabPanel, { children: /* @__PURE__ */ jsx(
573
- ReleasesGrid,
574
- {
575
- sectionTitle: "done",
576
- releases: response?.currentData?.data,
577
- isError
578
- }
579
- ) })
580
- ] })
581
- ]
582
- }
583
- ),
584
- /* @__PURE__ */ jsxs(
585
- Pagination.Root,
586
- {
587
- ...response?.currentData?.meta?.pagination,
588
- defaultPageSize: response?.currentData?.meta?.pagination?.pageSize,
589
- children: [
590
- /* @__PURE__ */ jsx(Pagination.PageSize, { options: ["8", "16", "32", "64"] }),
591
- /* @__PURE__ */ jsx(Pagination.Links, {})
592
- ]
593
- }
594
- )
595
- ] }) }),
596
- releaseModalShown && /* @__PURE__ */ jsx(
597
- ReleaseModal,
598
- {
599
- handleClose: toggleAddReleaseModal,
600
- handleSubmit: handleAddRelease,
601
- isLoading: isSubmittingForm,
602
- initialValues: INITIAL_FORM_VALUES
603
- }
604
- )
605
- ] });
606
- };
607
- const ReleaseInfoWrapper = styled(Flex)`
608
- align-self: stretch;
609
- border-bottom-right-radius: ${({ theme }) => theme.borderRadius};
610
- border-bottom-left-radius: ${({ theme }) => theme.borderRadius};
611
- border-top: 1px solid ${({ theme }) => theme.colors.neutral150};
612
- `;
613
- const StyledMenuItem = styled(Menu.Item)`
614
- svg path {
615
- fill: ${({ theme, disabled }) => disabled && theme.colors.neutral500};
616
- }
617
- span {
618
- color: ${({ theme, disabled }) => disabled && theme.colors.neutral500};
619
- }
620
-
621
- &:hover {
622
- background: ${({ theme, variant = "neutral" }) => theme.colors[`${variant}100`]};
623
- }
624
- `;
625
- const PencilIcon = styled(Pencil)`
626
- width: ${({ theme }) => theme.spaces[3]};
627
- height: ${({ theme }) => theme.spaces[3]};
628
- path {
629
- fill: ${({ theme }) => theme.colors.neutral600};
630
- }
631
- `;
632
- const TrashIcon = styled(Trash)`
633
- width: ${({ theme }) => theme.spaces[3]};
634
- height: ${({ theme }) => theme.spaces[3]};
635
- path {
636
- fill: ${({ theme }) => theme.colors.danger600};
637
- }
638
- `;
639
- const TypographyMaxWidth = styled(Typography)`
640
- max-width: 300px;
641
- `;
642
- const EntryValidationText = ({ action, schema, entry }) => {
643
- const { formatMessage } = useIntl();
644
- const { validate } = unstable_useDocument(
645
- {
646
- collectionType: schema?.kind ?? "",
647
- model: schema?.uid ?? ""
648
- },
649
- {
650
- skip: !schema
651
- }
652
- );
653
- const errors = validate(entry) ?? {};
654
- if (Object.keys(errors).length > 0) {
655
- const validationErrorsMessages = Object.entries(errors).map(
656
- ([key, value]) => formatMessage(
657
- { id: `${value.id}.withField`, defaultMessage: value.defaultMessage },
658
- { field: key }
659
- )
660
- ).join(" ");
661
- return /* @__PURE__ */ jsxs(Flex, { gap: 2, children: [
662
- /* @__PURE__ */ jsx(Icon, { color: "danger600", as: CrossCircle }),
663
- /* @__PURE__ */ jsx(Tooltip, { description: validationErrorsMessages, children: /* @__PURE__ */ jsx(TypographyMaxWidth, { textColor: "danger600", variant: "omega", fontWeight: "semiBold", ellipsis: true, children: validationErrorsMessages }) })
664
- ] });
665
- }
666
- if (action == "publish") {
667
- return /* @__PURE__ */ jsxs(Flex, { gap: 2, children: [
668
- /* @__PURE__ */ jsx(Icon, { color: "success600", as: CheckCircle }),
669
- entry.publishedAt ? /* @__PURE__ */ jsx(Typography, { textColor: "success600", fontWeight: "bold", children: formatMessage({
670
- id: "content-releases.pages.ReleaseDetails.entry-validation.already-published",
671
- defaultMessage: "Already published"
672
- }) }) : /* @__PURE__ */ jsx(Typography, { children: formatMessage({
673
- id: "content-releases.pages.ReleaseDetails.entry-validation.ready-to-publish",
674
- defaultMessage: "Ready to publish"
675
- }) })
676
- ] });
677
- }
678
- return /* @__PURE__ */ jsxs(Flex, { gap: 2, children: [
679
- /* @__PURE__ */ jsx(Icon, { color: "success600", as: CheckCircle }),
680
- !entry.publishedAt ? /* @__PURE__ */ jsx(Typography, { textColor: "success600", fontWeight: "bold", children: formatMessage({
681
- id: "content-releases.pages.ReleaseDetails.entry-validation.already-unpublished",
682
- defaultMessage: "Already unpublished"
683
- }) }) : /* @__PURE__ */ jsx(Typography, { children: formatMessage({
684
- id: "content-releases.pages.ReleaseDetails.entry-validation.ready-to-unpublish",
685
- defaultMessage: "Ready to unpublish"
686
- }) })
687
- ] });
688
- };
689
- const ReleaseDetailsLayout = ({
690
- toggleEditReleaseModal,
691
- toggleWarningSubmit,
692
- children
693
- }) => {
694
- const { formatMessage, formatDate, formatTime } = useIntl();
695
- const { releaseId } = useParams();
696
- const {
697
- data,
698
- isLoading: isLoadingDetails,
699
- isError,
700
- error
701
- } = useGetReleaseQuery(
702
- { id: releaseId },
703
- {
704
- skip: !releaseId
705
- }
706
- );
707
- const [publishRelease, { isLoading: isPublishing }] = usePublishReleaseMutation();
708
- const toggleNotification = useNotification();
709
- const { formatAPIError } = useAPIErrorHandler();
710
- const {
711
- allowedActions: { canUpdate, canDelete }
712
- } = useRBAC(PERMISSIONS);
713
- const dispatch = useTypedDispatch();
714
- const { trackUsage } = useTracking();
715
- const release = data?.data;
716
- const handlePublishRelease = (id) => async () => {
717
- const response = await publishRelease({ id });
718
- if ("data" in response) {
719
- toggleNotification({
720
- type: "success",
721
- message: formatMessage({
722
- id: "content-releases.pages.ReleaseDetails.publish-notification-success",
723
- defaultMessage: "Release was published successfully."
724
- })
725
- });
726
- const { totalEntries: totalEntries2, totalPublishedEntries, totalUnpublishedEntries } = response.data.meta;
727
- trackUsage("didPublishRelease", {
728
- totalEntries: totalEntries2,
729
- totalPublishedEntries,
730
- totalUnpublishedEntries
731
- });
732
- } else if (isAxiosError(response.error)) {
733
- toggleNotification({
734
- type: "warning",
735
- message: formatAPIError(response.error)
736
- });
737
- } else {
738
- toggleNotification({
739
- type: "warning",
740
- message: formatMessage({ id: "notification.error", defaultMessage: "An error occurred" })
741
- });
742
- }
743
- };
744
- const handleRefresh = () => {
745
- dispatch(releaseApi.util.invalidateTags([{ type: "ReleaseAction", id: "LIST" }]));
746
- };
747
- const getCreatedByUser = () => {
748
- if (!release?.createdBy) {
749
- return null;
750
- }
751
- if (release.createdBy.username) {
752
- return release.createdBy.username;
753
- }
754
- if (release.createdBy.firstname) {
755
- return `${release.createdBy.firstname} ${release.createdBy.lastname || ""}`.trim();
756
- }
757
- return release.createdBy.email;
758
- };
759
- if (isLoadingDetails) {
760
- return /* @__PURE__ */ jsx(Page.Loading, {});
761
- }
762
- if (isError || !release) {
763
- return /* @__PURE__ */ jsx(
764
- Navigate,
765
- {
766
- to: "..",
767
- state: {
768
- errors: [
769
- {
770
- code: error?.code
771
- }
772
- ]
773
- }
774
- }
775
- );
776
- }
777
- const totalEntries = release.actions.meta.count || 0;
778
- const hasCreatedByUser = Boolean(getCreatedByUser());
779
- const IsSchedulingEnabled = window.strapi.future.isEnabled("contentReleasesScheduling");
780
- const isScheduled = release.scheduledAt && release.timezone;
781
- const numberOfEntriesText = formatMessage(
782
- {
783
- id: "content-releases.pages.Details.header-subtitle",
784
- defaultMessage: "{number, plural, =0 {No entries} one {# entry} other {# entries}}"
785
- },
786
- { number: totalEntries }
787
- );
788
- const scheduledText = isScheduled ? formatMessage(
789
- {
790
- id: "content-releases.pages.ReleaseDetails.header-subtitle.scheduled",
791
- defaultMessage: "Scheduled for {date} at {time} ({offset})"
792
- },
793
- {
794
- date: formatDate(new Date(release.scheduledAt), {
795
- weekday: "long",
796
- day: "numeric",
797
- month: "long",
798
- year: "numeric",
799
- timeZone: release.timezone
800
- }),
801
- time: formatTime(new Date(release.scheduledAt), {
802
- timeZone: release.timezone,
803
- hourCycle: "h23"
804
- }),
805
- offset: getTimezoneOffset(release.timezone, new Date(release.scheduledAt))
806
- }
807
- ) : "";
808
- return /* @__PURE__ */ jsxs(Main, { "aria-busy": isLoadingDetails, children: [
809
- /* @__PURE__ */ jsx(
810
- HeaderLayout,
811
- {
812
- title: release.name,
813
- subtitle: /* @__PURE__ */ jsxs(Flex, { gap: 2, lineHeight: 6, children: [
814
- /* @__PURE__ */ jsx(Typography, { textColor: "neutral600", variant: "epsilon", children: numberOfEntriesText + (IsSchedulingEnabled && isScheduled ? ` - ${scheduledText}` : "") }),
815
- /* @__PURE__ */ jsx(Badge, { ...getBadgeProps(release.status), children: release.status })
816
- ] }),
817
- navigationAction: /* @__PURE__ */ jsx(BackButton, {}),
818
- primaryAction: !release.releasedAt && /* @__PURE__ */ jsxs(Flex, { gap: 2, children: [
819
- /* @__PURE__ */ jsxs(Menu.Root, { children: [
820
- /* @__PURE__ */ jsx(
821
- Menu.Trigger,
822
- {
823
- as: IconButton,
824
- paddingLeft: 2,
825
- paddingRight: 2,
826
- "aria-label": formatMessage({
827
- id: "content-releases.header.actions.open-release-actions",
828
- defaultMessage: "Release edit and delete menu"
829
- }),
830
- icon: /* @__PURE__ */ jsx(More, {}),
831
- variant: "tertiary"
832
- }
833
- ),
834
- /* @__PURE__ */ jsxs(Menu.Content, { top: 1, popoverPlacement: "bottom-end", children: [
835
- /* @__PURE__ */ jsxs(
836
- Flex,
837
- {
838
- alignItems: "center",
839
- justifyContent: "center",
840
- direction: "column",
841
- padding: 1,
842
- width: "100%",
843
- children: [
844
- /* @__PURE__ */ jsx(StyledMenuItem, { disabled: !canUpdate, onSelect: toggleEditReleaseModal, children: /* @__PURE__ */ jsxs(Flex, { alignItems: "center", gap: 2, hasRadius: true, width: "100%", children: [
845
- /* @__PURE__ */ jsx(PencilIcon, {}),
846
- /* @__PURE__ */ jsx(Typography, { ellipsis: true, children: formatMessage({
847
- id: "content-releases.header.actions.edit",
848
- defaultMessage: "Edit"
849
- }) })
850
- ] }) }),
851
- /* @__PURE__ */ jsx(
852
- StyledMenuItem,
853
- {
854
- disabled: !canDelete,
855
- onSelect: toggleWarningSubmit,
856
- variant: "danger",
857
- children: /* @__PURE__ */ jsxs(Flex, { alignItems: "center", gap: 2, hasRadius: true, width: "100%", children: [
858
- /* @__PURE__ */ jsx(TrashIcon, {}),
859
- /* @__PURE__ */ jsx(Typography, { ellipsis: true, textColor: "danger600", children: formatMessage({
860
- id: "content-releases.header.actions.delete",
861
- defaultMessage: "Delete"
862
- }) })
863
- ] })
864
- }
865
- )
866
- ]
867
- }
868
- ),
869
- /* @__PURE__ */ jsxs(
870
- ReleaseInfoWrapper,
871
- {
872
- direction: "column",
873
- justifyContent: "center",
874
- alignItems: "flex-start",
875
- gap: 1,
876
- padding: 5,
877
- children: [
878
- /* @__PURE__ */ jsx(Typography, { variant: "pi", fontWeight: "bold", children: formatMessage({
879
- id: "content-releases.header.actions.created",
880
- defaultMessage: "Created"
881
- }) }),
882
- /* @__PURE__ */ jsxs(Typography, { variant: "pi", color: "neutral300", children: [
883
- /* @__PURE__ */ jsx(RelativeTime, { timestamp: new Date(release.createdAt) }),
884
- formatMessage(
885
- {
886
- id: "content-releases.header.actions.created.description",
887
- defaultMessage: "{hasCreatedByUser, select, true { by {createdBy}} other { by deleted user}}"
888
- },
889
- { createdBy: getCreatedByUser(), hasCreatedByUser }
890
- )
891
- ] })
892
- ]
893
- }
894
- )
895
- ] })
896
- ] }),
897
- /* @__PURE__ */ jsx(Button, { size: "S", variant: "tertiary", onClick: handleRefresh, children: formatMessage({
898
- id: "content-releases.header.actions.refresh",
899
- defaultMessage: "Refresh"
900
- }) }),
901
- /* @__PURE__ */ jsx(CheckPermissions, { permissions: PERMISSIONS.publish, children: /* @__PURE__ */ jsx(
902
- Button,
903
- {
904
- size: "S",
905
- variant: "default",
906
- onClick: handlePublishRelease(release.id.toString()),
907
- loading: isPublishing,
908
- disabled: release.actions.meta.count === 0,
909
- children: formatMessage({
910
- id: "content-releases.header.actions.publish",
911
- defaultMessage: "Publish"
912
- })
913
- }
914
- ) })
915
- ] })
916
- }
917
- ),
918
- children
919
- ] });
920
- };
921
- const GROUP_BY_OPTIONS = ["contentType", "locale", "action"];
922
- const getGroupByOptionLabel = (value) => {
923
- if (value === "locale") {
924
- return {
925
- id: "content-releases.pages.ReleaseDetails.groupBy.option.locales",
926
- defaultMessage: "Locales"
927
- };
928
- }
929
- if (value === "action") {
930
- return {
931
- id: "content-releases.pages.ReleaseDetails.groupBy.option.actions",
932
- defaultMessage: "Actions"
933
- };
934
- }
935
- return {
936
- id: "content-releases.pages.ReleaseDetails.groupBy.option.content-type",
937
- defaultMessage: "Content-Types"
938
- };
939
- };
940
- const ReleaseDetailsBody = ({ releaseId }) => {
941
- const { formatMessage } = useIntl();
942
- const [{ query }, setQuery] = useQueryParams();
943
- const toggleNotification = useNotification();
944
- const { formatAPIError } = useAPIErrorHandler();
945
- const {
946
- data: releaseData,
947
- isLoading: isReleaseLoading,
948
- isError: isReleaseError,
949
- error: releaseError
950
- } = useGetReleaseQuery({ id: releaseId });
951
- const {
952
- allowedActions: { canUpdate }
953
- } = useRBAC(PERMISSIONS);
954
- const release = releaseData?.data;
955
- const selectedGroupBy = query?.groupBy || "contentType";
956
- const {
957
- isLoading,
958
- isFetching,
959
- isError,
960
- data,
961
- error: releaseActionsError
962
- } = useGetReleaseActionsQuery({
963
- ...query,
964
- releaseId
965
- });
966
- const [updateReleaseAction] = useUpdateReleaseActionMutation();
967
- const handleChangeType = async (e, actionId, actionPath) => {
968
- const response = await updateReleaseAction({
969
- params: {
970
- releaseId,
971
- actionId
972
- },
973
- body: {
974
- type: e.target.value
975
- },
976
- query,
977
- // We are passing the query params to make optimistic updates
978
- actionPath
979
- // We are passing the action path to found the position in the cache of the action for optimistic updates
980
- });
981
- if ("error" in response) {
982
- if (isAxiosError(response.error)) {
983
- toggleNotification({
984
- type: "warning",
985
- message: formatAPIError(response.error)
986
- });
987
- } else {
988
- toggleNotification({
989
- type: "warning",
990
- message: formatMessage({ id: "notification.error", defaultMessage: "An error occurred" })
991
- });
992
- }
993
- }
994
- };
995
- if (isLoading || isReleaseLoading) {
996
- return /* @__PURE__ */ jsx(Page.Loading, {});
997
- }
998
- const releaseActions = data?.data;
999
- const releaseMeta = data?.meta;
1000
- const contentTypes = releaseMeta?.contentTypes || {};
1001
- const components = releaseMeta?.components || {};
1002
- if (isReleaseError || !release) {
1003
- const errorsArray = [];
1004
- if (releaseError) {
1005
- errorsArray.push({
1006
- code: releaseError.code
1007
- });
1008
- }
1009
- if (releaseActionsError) {
1010
- errorsArray.push({
1011
- code: releaseActionsError.code
1012
- });
1013
- }
1014
- return /* @__PURE__ */ jsx(
1015
- Navigate,
1016
- {
1017
- to: "..",
1018
- state: {
1019
- errors: errorsArray
1020
- }
1021
- }
1022
- );
1023
- }
1024
- if (isError || !releaseActions) {
1025
- return /* @__PURE__ */ jsx(Page.Error, {});
1026
- }
1027
- if (Object.keys(releaseActions).length === 0) {
1028
- return /* @__PURE__ */ jsx(ContentLayout, { children: /* @__PURE__ */ jsx(
1029
- EmptyStateLayout,
1030
- {
1031
- action: /* @__PURE__ */ jsx(
1032
- LinkButton,
1033
- {
1034
- as: Link$1,
1035
- to: {
1036
- pathname: "/content-manager"
1037
- },
1038
- style: { textDecoration: "none" },
1039
- variant: "secondary",
1040
- children: formatMessage({
1041
- id: "content-releases.page.Details.button.openContentManager",
1042
- defaultMessage: "Open the Content Manager"
1043
- })
1044
- }
1045
- ),
1046
- icon: /* @__PURE__ */ jsx(EmptyDocuments, { width: "10rem" }),
1047
- content: formatMessage({
1048
- id: "content-releases.pages.Details.tab.emptyEntries",
1049
- defaultMessage: "This release is empty. Open the Content Manager, select an entry and add it to the release."
1050
- })
1051
- }
1052
- ) });
1053
- }
1054
- const headers = [
1055
- {
1056
- label: formatMessage({
1057
- id: "content-releases.page.ReleaseDetails.table.header.label.name",
1058
- defaultMessage: "name"
1059
- }),
1060
- name: "name"
1061
- },
1062
- {
1063
- label: formatMessage({
1064
- id: "content-releases.page.ReleaseDetails.table.header.label.locale",
1065
- defaultMessage: "locale"
1066
- }),
1067
- name: "locale"
1068
- },
1069
- {
1070
- label: formatMessage({
1071
- id: "content-releases.page.ReleaseDetails.table.header.label.content-type",
1072
- defaultMessage: "content-type"
1073
- }),
1074
- name: "content-type"
1075
- },
1076
- {
1077
- label: formatMessage({
1078
- id: "content-releases.page.ReleaseDetails.table.header.label.action",
1079
- defaultMessage: "action"
1080
- }),
1081
- name: "action"
1082
- },
1083
- ...!release.releasedAt ? [
1084
- {
1085
- label: formatMessage({
1086
- id: "content-releases.page.ReleaseDetails.table.header.label.status",
1087
- defaultMessage: "status"
1088
- }),
1089
- name: "status"
1090
- }
1091
- ] : []
1092
- ];
1093
- return /* @__PURE__ */ jsx(ContentLayout, { children: /* @__PURE__ */ jsxs(Flex, { gap: 8, direction: "column", alignItems: "stretch", children: [
1094
- /* @__PURE__ */ jsx(Flex, { children: /* @__PURE__ */ jsx(
1095
- SingleSelect,
1096
- {
1097
- "aria-label": formatMessage({
1098
- id: "content-releases.pages.ReleaseDetails.groupBy.aria-label",
1099
- defaultMessage: "Group by"
1100
- }),
1101
- customizeContent: (value) => formatMessage(
1102
- {
1103
- id: `content-releases.pages.ReleaseDetails.groupBy.label`,
1104
- defaultMessage: `Group by {groupBy}`
1105
- },
1106
- {
1107
- groupBy: value
1108
- }
1109
- ),
1110
- value: formatMessage(getGroupByOptionLabel(selectedGroupBy)),
1111
- onChange: (value) => setQuery({ groupBy: value }),
1112
- children: GROUP_BY_OPTIONS.map((option) => /* @__PURE__ */ jsx(SingleSelectOption, { value: option, children: formatMessage(getGroupByOptionLabel(option)) }, option))
1113
- }
1114
- ) }),
1115
- Object.keys(releaseActions).map((key) => /* @__PURE__ */ jsxs(Flex, { gap: 4, direction: "column", alignItems: "stretch", children: [
1116
- /* @__PURE__ */ jsx(Flex, { role: "separator", "aria-label": key, children: /* @__PURE__ */ jsx(Badge, { children: key }) }),
1117
- /* @__PURE__ */ jsx(
1118
- Table.Root,
1119
- {
1120
- rows: releaseActions[key].map((item) => ({
1121
- ...item,
1122
- id: Number(item.entry.id)
1123
- })),
1124
- headers,
1125
- isLoading: isLoading || isFetching,
1126
- children: /* @__PURE__ */ jsxs(Table.Content, { children: [
1127
- /* @__PURE__ */ jsx(Table.Head, { children: headers.map((header) => /* @__PURE__ */ jsx(Table.HeaderCell, { ...header }, header.name)) }),
1128
- /* @__PURE__ */ jsx(Table.Loading, {}),
1129
- /* @__PURE__ */ jsx(Table.Body, { children: releaseActions[key].map(
1130
- ({ id, contentType, locale, type, entry }, actionIndex) => /* @__PURE__ */ jsxs(Tr, { children: [
1131
- /* @__PURE__ */ jsx(Td, { width: "25%", maxWidth: "200px", children: /* @__PURE__ */ jsx(Typography, { ellipsis: true, children: `${contentType.mainFieldValue || entry.id}` }) }),
1132
- /* @__PURE__ */ jsx(Td, { width: "10%", children: /* @__PURE__ */ jsx(Typography, { children: `${locale?.name ? locale.name : "-"}` }) }),
1133
- /* @__PURE__ */ jsx(Td, { width: "10%", children: /* @__PURE__ */ jsx(Typography, { children: contentType.displayName || "" }) }),
1134
- /* @__PURE__ */ jsx(Td, { width: "20%", children: release.releasedAt ? /* @__PURE__ */ jsx(Typography, { children: formatMessage(
1135
- {
1136
- id: "content-releases.page.ReleaseDetails.table.action-published",
1137
- defaultMessage: "This entry was <b>{isPublish, select, true {published} other {unpublished}}</b>."
1138
- },
1139
- {
1140
- isPublish: type === "publish",
1141
- b: (children) => /* @__PURE__ */ jsx(Typography, { fontWeight: "bold", children })
1142
- }
1143
- ) }) : /* @__PURE__ */ jsx(
1144
- ReleaseActionOptions,
1145
- {
1146
- selected: type,
1147
- handleChange: (e) => handleChangeType(e, id, [key, actionIndex]),
1148
- name: `release-action-${id}-type`,
1149
- disabled: !canUpdate
1150
- }
1151
- ) }),
1152
- !release.releasedAt && /* @__PURE__ */ jsxs(Fragment, { children: [
1153
- /* @__PURE__ */ jsx(Td, { width: "20%", minWidth: "200px", children: /* @__PURE__ */ jsx(
1154
- EntryValidationText,
1155
- {
1156
- action: type,
1157
- schema: contentTypes?.[contentType.uid],
1158
- components,
1159
- entry
1160
- }
1161
- ) }),
1162
- /* @__PURE__ */ jsx(Td, { children: /* @__PURE__ */ jsx(Flex, { justifyContent: "flex-end", children: /* @__PURE__ */ jsxs(ReleaseActionMenu.Root, { children: [
1163
- /* @__PURE__ */ jsx(
1164
- ReleaseActionMenu.ReleaseActionEntryLinkItem,
1165
- {
1166
- contentTypeUid: contentType.uid,
1167
- entryId: entry.id,
1168
- locale: locale?.code
1169
- }
1170
- ),
1171
- /* @__PURE__ */ jsx(
1172
- ReleaseActionMenu.DeleteReleaseActionItem,
1173
- {
1174
- releaseId: release.id,
1175
- actionId: id
1176
- }
1177
- )
1178
- ] }) }) })
1179
- ] })
1180
- ] }, id)
1181
- ) })
1182
- ] })
1183
- }
1184
- )
1185
- ] }, `releases-group-${key}`)),
1186
- /* @__PURE__ */ jsxs(
1187
- Pagination.Root,
1188
- {
1189
- ...releaseMeta?.pagination,
1190
- defaultPageSize: releaseMeta?.pagination?.pageSize,
1191
- children: [
1192
- /* @__PURE__ */ jsx(Pagination.PageSize, {}),
1193
- /* @__PURE__ */ jsx(Pagination.Links, {})
1194
- ]
1195
- }
1196
- )
1197
- ] }) });
1198
- };
1199
- const ReleaseDetailsPage = () => {
1200
- const { formatMessage } = useIntl();
1201
- const { releaseId } = useParams();
1202
- const toggleNotification = useNotification();
1203
- const { formatAPIError } = useAPIErrorHandler();
1204
- const navigate = useNavigate();
1205
- const [releaseModalShown, setReleaseModalShown] = React.useState(false);
1206
- const [showWarningSubmit, setWarningSubmit] = React.useState(false);
1207
- const {
1208
- isLoading: isLoadingDetails,
1209
- data,
1210
- isSuccess: isSuccessDetails
1211
- } = useGetReleaseQuery(
1212
- { id: releaseId },
1213
- {
1214
- skip: !releaseId
1215
- }
1216
- );
1217
- const [updateRelease, { isLoading: isSubmittingForm }] = useUpdateReleaseMutation();
1218
- const [deleteRelease, { isLoading: isDeletingRelease }] = useDeleteReleaseMutation();
1219
- const toggleEditReleaseModal = () => {
1220
- setReleaseModalShown((prev) => !prev);
1221
- };
1222
- const toggleWarningSubmit = () => setWarningSubmit((prevState) => !prevState);
1223
- if (isLoadingDetails) {
1224
- return /* @__PURE__ */ jsx(
1225
- ReleaseDetailsLayout,
1226
- {
1227
- toggleEditReleaseModal,
1228
- toggleWarningSubmit,
1229
- children: /* @__PURE__ */ jsx(Page.Loading, {})
1230
- }
1231
- );
1232
- }
1233
- if (!releaseId) {
1234
- return /* @__PURE__ */ jsx(Navigate, { to: ".." });
1235
- }
1236
- const releaseData = isSuccessDetails && data?.data || null;
1237
- const title = releaseData?.name || "";
1238
- const timezone = releaseData?.timezone ?? null;
1239
- const scheduledAt = releaseData?.scheduledAt && timezone ? utcToZonedTime(releaseData.scheduledAt, timezone) : null;
1240
- const date = scheduledAt ? new Date(format(scheduledAt, "yyyy-MM-dd")) : null;
1241
- const time = scheduledAt ? format(scheduledAt, "HH:mm") : "";
1242
- const handleEditRelease = async (values) => {
1243
- const response = await updateRelease({
1244
- id: releaseId,
1245
- name: values.name,
1246
- scheduledAt: values.scheduledAt,
1247
- timezone: values.timezone
1248
- });
1249
- if ("data" in response) {
1250
- toggleNotification({
1251
- type: "success",
1252
- message: formatMessage({
1253
- id: "content-releases.modal.release-updated-notification-success",
1254
- defaultMessage: "Release updated."
1255
- })
1256
- });
1257
- } else if (isAxiosError(response.error)) {
1258
- toggleNotification({
1259
- type: "warning",
1260
- message: formatAPIError(response.error)
1261
- });
1262
- } else {
1263
- toggleNotification({
1264
- type: "warning",
1265
- message: formatMessage({ id: "notification.error", defaultMessage: "An error occurred" })
1266
- });
1267
- }
1268
- toggleEditReleaseModal();
1269
- };
1270
- const handleDeleteRelease = async () => {
1271
- const response = await deleteRelease({
1272
- id: releaseId
1273
- });
1274
- if ("data" in response) {
1275
- navigate("..");
1276
- } else if (isAxiosError(response.error)) {
1277
- toggleNotification({
1278
- type: "warning",
1279
- message: formatAPIError(response.error)
1280
- });
1281
- } else {
1282
- toggleNotification({
1283
- type: "warning",
1284
- message: formatMessage({ id: "notification.error", defaultMessage: "An error occurred" })
1285
- });
1286
- }
1287
- };
1288
- return /* @__PURE__ */ jsxs(
1289
- ReleaseDetailsLayout,
1290
- {
1291
- toggleEditReleaseModal,
1292
- toggleWarningSubmit,
1293
- children: [
1294
- /* @__PURE__ */ jsx(ReleaseDetailsBody, { releaseId }),
1295
- releaseModalShown && /* @__PURE__ */ jsx(
1296
- ReleaseModal,
1297
- {
1298
- handleClose: toggleEditReleaseModal,
1299
- handleSubmit: handleEditRelease,
1300
- isLoading: isLoadingDetails || isSubmittingForm,
1301
- initialValues: {
1302
- name: title || "",
1303
- scheduledAt,
1304
- date,
1305
- time,
1306
- isScheduled: Boolean(scheduledAt),
1307
- timezone
1308
- }
1309
- }
1310
- ),
1311
- /* @__PURE__ */ jsx(
1312
- ConfirmDialog,
1313
- {
1314
- bodyText: {
1315
- id: "content-releases.dialog.confirmation-message",
1316
- defaultMessage: "Are you sure you want to delete this release?"
1317
- },
1318
- isOpen: showWarningSubmit,
1319
- isConfirmButtonLoading: isDeletingRelease,
1320
- onToggleDialog: toggleWarningSubmit,
1321
- onConfirm: handleDeleteRelease
1322
- }
1323
- )
1324
- ]
1325
- }
1326
- );
1327
- };
1328
- const App = () => {
1329
- return /* @__PURE__ */ jsx(Page.Protect, { permissions: PERMISSIONS.main, children: /* @__PURE__ */ jsxs(Routes, { children: [
1330
- /* @__PURE__ */ jsx(Route, { index: true, element: /* @__PURE__ */ jsx(ReleasesPage, {}) }),
1331
- /* @__PURE__ */ jsx(Route, { path: ":releaseId", element: /* @__PURE__ */ jsx(ReleaseDetailsPage, {}) })
1332
- ] }) });
1333
- };
1334
- export {
1335
- App
1336
- };
1337
- //# sourceMappingURL=App-PQlYzNfb.mjs.map