@strapi/content-releases 0.0.0-experimental.fc1ac2acd58c8a5a858679956b6d102ac5ee4011 → 0.0.0-experimental.fca58d9b01ba89012d84a58a20556a6fd716cb92

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 +243 -0
  23. package/dist/admin/components/ReleaseActionMenu.js.map +1 -0
  24. package/dist/admin/components/ReleaseActionMenu.mjs +222 -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 +821 -0
  67. package/dist/admin/pages/ReleaseDetailsPage.js.map +1 -0
  68. package/dist/admin/pages/ReleaseDetailsPage.mjs +800 -0
  69. package/dist/admin/pages/ReleaseDetailsPage.mjs.map +1 -0
  70. package/dist/admin/pages/ReleasesPage.js +397 -0
  71. package/dist/admin/pages/ReleasesPage.js.map +1 -0
  72. package/dist/admin/pages/ReleasesPage.mjs +375 -0
  73. package/dist/admin/pages/ReleasesPage.mjs.map +1 -0
  74. package/dist/admin/pages/ReleasesSettingsPage.js +199 -0
  75. package/dist/admin/pages/ReleasesSettingsPage.js.map +1 -0
  76. package/dist/admin/pages/ReleasesSettingsPage.mjs +197 -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 -1569
  190. package/dist/server/index.js.map +1 -1
  191. package/dist/server/index.mjs +24 -1549
  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 +1514 -3241
  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 +1478 -3226
  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 +19 -1800
  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 -16
  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-0Er6xxcq.mjs +0 -1358
  319. package/dist/_chunks/App-0Er6xxcq.mjs.map +0 -1
  320. package/dist/_chunks/App-C768ulk4.js +0 -1381
  321. package/dist/_chunks/App-C768ulk4.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-BLSMpbpZ.js +0 -1028
  331. package/dist/_chunks/index-BLSMpbpZ.js.map +0 -1
  332. package/dist/_chunks/index-fJx1up7m.mjs +0 -1007
  333. package/dist/_chunks/index-fJx1up7m.mjs.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,1358 +0,0 @@
1
- import { jsx, jsxs, Fragment } from "react/jsx-runtime";
2
- import { useLicenseLimits, Page, Pagination, unstable_useDocument } from "@strapi/admin/strapi-admin";
3
- import { useLocation, useNavigate, useParams, Navigate, Link as Link$2, 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-fJx1up7m.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, Link as Link$1, 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, Table } from "@strapi/helper-plugin";
9
- import { Plus, EmptyDocuments, Pencil, Trash, ArrowLeft, 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(Link$1, { startIcon: /* @__PURE__ */ jsx(ArrowLeft, {}), to: "/plugins/content-releases", children: formatMessage({
818
- id: "global.back",
819
- defaultMessage: "Back"
820
- }) }),
821
- primaryAction: !release.releasedAt && /* @__PURE__ */ jsxs(Flex, { gap: 2, children: [
822
- /* @__PURE__ */ jsxs(Menu.Root, { children: [
823
- /* @__PURE__ */ jsx(
824
- Menu.Trigger,
825
- {
826
- as: IconButton,
827
- paddingLeft: 2,
828
- paddingRight: 2,
829
- "aria-label": formatMessage({
830
- id: "content-releases.header.actions.open-release-actions",
831
- defaultMessage: "Release edit and delete menu"
832
- }),
833
- icon: /* @__PURE__ */ jsx(More, {}),
834
- variant: "tertiary"
835
- }
836
- ),
837
- /* @__PURE__ */ jsxs(Menu.Content, { top: 1, popoverPlacement: "bottom-end", children: [
838
- /* @__PURE__ */ jsxs(
839
- Flex,
840
- {
841
- alignItems: "center",
842
- justifyContent: "center",
843
- direction: "column",
844
- padding: 1,
845
- width: "100%",
846
- children: [
847
- /* @__PURE__ */ jsx(StyledMenuItem, { disabled: !canUpdate, onSelect: toggleEditReleaseModal, children: /* @__PURE__ */ jsxs(Flex, { alignItems: "center", gap: 2, hasRadius: true, width: "100%", children: [
848
- /* @__PURE__ */ jsx(PencilIcon, {}),
849
- /* @__PURE__ */ jsx(Typography, { ellipsis: true, children: formatMessage({
850
- id: "content-releases.header.actions.edit",
851
- defaultMessage: "Edit"
852
- }) })
853
- ] }) }),
854
- /* @__PURE__ */ jsx(
855
- StyledMenuItem,
856
- {
857
- disabled: !canDelete,
858
- onSelect: toggleWarningSubmit,
859
- variant: "danger",
860
- children: /* @__PURE__ */ jsxs(Flex, { alignItems: "center", gap: 2, hasRadius: true, width: "100%", children: [
861
- /* @__PURE__ */ jsx(TrashIcon, {}),
862
- /* @__PURE__ */ jsx(Typography, { ellipsis: true, textColor: "danger600", children: formatMessage({
863
- id: "content-releases.header.actions.delete",
864
- defaultMessage: "Delete"
865
- }) })
866
- ] })
867
- }
868
- )
869
- ]
870
- }
871
- ),
872
- /* @__PURE__ */ jsxs(
873
- ReleaseInfoWrapper,
874
- {
875
- direction: "column",
876
- justifyContent: "center",
877
- alignItems: "flex-start",
878
- gap: 1,
879
- padding: 5,
880
- children: [
881
- /* @__PURE__ */ jsx(Typography, { variant: "pi", fontWeight: "bold", children: formatMessage({
882
- id: "content-releases.header.actions.created",
883
- defaultMessage: "Created"
884
- }) }),
885
- /* @__PURE__ */ jsxs(Typography, { variant: "pi", color: "neutral300", children: [
886
- /* @__PURE__ */ jsx(RelativeTime, { timestamp: new Date(release.createdAt) }),
887
- formatMessage(
888
- {
889
- id: "content-releases.header.actions.created.description",
890
- defaultMessage: "{hasCreatedByUser, select, true { by {createdBy}} other { by deleted user}}"
891
- },
892
- { createdBy: getCreatedByUser(), hasCreatedByUser }
893
- )
894
- ] })
895
- ]
896
- }
897
- )
898
- ] })
899
- ] }),
900
- /* @__PURE__ */ jsx(Button, { size: "S", variant: "tertiary", onClick: handleRefresh, children: formatMessage({
901
- id: "content-releases.header.actions.refresh",
902
- defaultMessage: "Refresh"
903
- }) }),
904
- /* @__PURE__ */ jsx(CheckPermissions, { permissions: PERMISSIONS.publish, children: /* @__PURE__ */ jsx(
905
- Button,
906
- {
907
- size: "S",
908
- variant: "default",
909
- onClick: handlePublishRelease(release.id.toString()),
910
- loading: isPublishing,
911
- disabled: release.actions.meta.count === 0,
912
- children: formatMessage({
913
- id: "content-releases.header.actions.publish",
914
- defaultMessage: "Publish"
915
- })
916
- }
917
- ) })
918
- ] })
919
- }
920
- ),
921
- children
922
- ] });
923
- };
924
- const GROUP_BY_OPTIONS = ["contentType", "locale", "action"];
925
- const getGroupByOptionLabel = (value) => {
926
- if (value === "locale") {
927
- return {
928
- id: "content-releases.pages.ReleaseDetails.groupBy.option.locales",
929
- defaultMessage: "Locales"
930
- };
931
- }
932
- if (value === "action") {
933
- return {
934
- id: "content-releases.pages.ReleaseDetails.groupBy.option.actions",
935
- defaultMessage: "Actions"
936
- };
937
- }
938
- return {
939
- id: "content-releases.pages.ReleaseDetails.groupBy.option.content-type",
940
- defaultMessage: "Content-Types"
941
- };
942
- };
943
- const ReleaseDetailsBody = ({ releaseId }) => {
944
- const { formatMessage } = useIntl();
945
- const [{ query }, setQuery] = useQueryParams();
946
- const toggleNotification = useNotification();
947
- const { formatAPIError } = useAPIErrorHandler();
948
- const {
949
- data: releaseData,
950
- isLoading: isReleaseLoading,
951
- isError: isReleaseError,
952
- error: releaseError
953
- } = useGetReleaseQuery({ id: releaseId });
954
- const {
955
- allowedActions: { canUpdate }
956
- } = useRBAC(PERMISSIONS);
957
- const release = releaseData?.data;
958
- const selectedGroupBy = query?.groupBy || "contentType";
959
- const {
960
- isLoading,
961
- isFetching,
962
- isError,
963
- data,
964
- error: releaseActionsError
965
- } = useGetReleaseActionsQuery({
966
- ...query,
967
- releaseId
968
- });
969
- const [updateReleaseAction] = useUpdateReleaseActionMutation();
970
- const handleChangeType = async (e, actionId, actionPath) => {
971
- const response = await updateReleaseAction({
972
- params: {
973
- releaseId,
974
- actionId
975
- },
976
- body: {
977
- type: e.target.value
978
- },
979
- query,
980
- // We are passing the query params to make optimistic updates
981
- actionPath
982
- // We are passing the action path to found the position in the cache of the action for optimistic updates
983
- });
984
- if ("error" in response) {
985
- if (isAxiosError(response.error)) {
986
- toggleNotification({
987
- type: "warning",
988
- message: formatAPIError(response.error)
989
- });
990
- } else {
991
- toggleNotification({
992
- type: "warning",
993
- message: formatMessage({ id: "notification.error", defaultMessage: "An error occurred" })
994
- });
995
- }
996
- }
997
- };
998
- if (isLoading || isReleaseLoading) {
999
- return /* @__PURE__ */ jsx(Page.Loading, {});
1000
- }
1001
- const releaseActions = data?.data;
1002
- const releaseMeta = data?.meta;
1003
- const contentTypes = releaseMeta?.contentTypes || {};
1004
- const components = releaseMeta?.components || {};
1005
- if (isReleaseError || !release) {
1006
- const errorsArray = [];
1007
- if (releaseError) {
1008
- errorsArray.push({
1009
- code: releaseError.code
1010
- });
1011
- }
1012
- if (releaseActionsError) {
1013
- errorsArray.push({
1014
- code: releaseActionsError.code
1015
- });
1016
- }
1017
- return /* @__PURE__ */ jsx(
1018
- Navigate,
1019
- {
1020
- to: "..",
1021
- state: {
1022
- errors: errorsArray
1023
- }
1024
- }
1025
- );
1026
- }
1027
- if (isError || !releaseActions) {
1028
- return /* @__PURE__ */ jsx(Page.Error, {});
1029
- }
1030
- if (Object.keys(releaseActions).length === 0) {
1031
- return /* @__PURE__ */ jsx(ContentLayout, { children: /* @__PURE__ */ jsx(
1032
- EmptyStateLayout,
1033
- {
1034
- action: /* @__PURE__ */ jsx(
1035
- LinkButton,
1036
- {
1037
- as: Link$2,
1038
- to: {
1039
- pathname: "/content-manager"
1040
- },
1041
- style: { textDecoration: "none" },
1042
- variant: "secondary",
1043
- children: formatMessage({
1044
- id: "content-releases.page.Details.button.openContentManager",
1045
- defaultMessage: "Open the Content Manager"
1046
- })
1047
- }
1048
- ),
1049
- icon: /* @__PURE__ */ jsx(EmptyDocuments, { width: "10rem" }),
1050
- content: formatMessage({
1051
- id: "content-releases.pages.Details.tab.emptyEntries",
1052
- defaultMessage: "This release is empty. Open the Content Manager, select an entry and add it to the release."
1053
- })
1054
- }
1055
- ) });
1056
- }
1057
- return /* @__PURE__ */ jsx(ContentLayout, { children: /* @__PURE__ */ jsxs(Flex, { gap: 8, direction: "column", alignItems: "stretch", children: [
1058
- /* @__PURE__ */ jsx(Flex, { children: /* @__PURE__ */ jsx(
1059
- SingleSelect,
1060
- {
1061
- "aria-label": formatMessage({
1062
- id: "content-releases.pages.ReleaseDetails.groupBy.aria-label",
1063
- defaultMessage: "Group by"
1064
- }),
1065
- customizeContent: (value) => formatMessage(
1066
- {
1067
- id: `content-releases.pages.ReleaseDetails.groupBy.label`,
1068
- defaultMessage: `Group by {groupBy}`
1069
- },
1070
- {
1071
- groupBy: value
1072
- }
1073
- ),
1074
- value: formatMessage(getGroupByOptionLabel(selectedGroupBy)),
1075
- onChange: (value) => setQuery({ groupBy: value }),
1076
- children: GROUP_BY_OPTIONS.map((option) => /* @__PURE__ */ jsx(SingleSelectOption, { value: option, children: formatMessage(getGroupByOptionLabel(option)) }, option))
1077
- }
1078
- ) }),
1079
- Object.keys(releaseActions).map((key) => /* @__PURE__ */ jsxs(Flex, { gap: 4, direction: "column", alignItems: "stretch", children: [
1080
- /* @__PURE__ */ jsx(Flex, { role: "separator", "aria-label": key, children: /* @__PURE__ */ jsx(Badge, { children: key }) }),
1081
- /* @__PURE__ */ jsx(
1082
- Table.Root,
1083
- {
1084
- rows: releaseActions[key].map((item) => ({
1085
- ...item,
1086
- id: Number(item.entry.id)
1087
- })),
1088
- colCount: releaseActions[key].length,
1089
- isLoading,
1090
- isFetching,
1091
- children: /* @__PURE__ */ jsxs(Table.Content, { children: [
1092
- /* @__PURE__ */ jsxs(Table.Head, { children: [
1093
- /* @__PURE__ */ jsx(
1094
- Table.HeaderCell,
1095
- {
1096
- attribute: { type: "string" },
1097
- label: formatMessage({
1098
- id: "content-releases.page.ReleaseDetails.table.header.label.name",
1099
- defaultMessage: "name"
1100
- }),
1101
- name: "name"
1102
- }
1103
- ),
1104
- /* @__PURE__ */ jsx(
1105
- Table.HeaderCell,
1106
- {
1107
- attribute: { type: "string" },
1108
- label: formatMessage({
1109
- id: "content-releases.page.ReleaseDetails.table.header.label.locale",
1110
- defaultMessage: "locale"
1111
- }),
1112
- name: "locale"
1113
- }
1114
- ),
1115
- /* @__PURE__ */ jsx(
1116
- Table.HeaderCell,
1117
- {
1118
- attribute: { type: "string" },
1119
- label: formatMessage({
1120
- id: "content-releases.page.ReleaseDetails.table.header.label.content-type",
1121
- defaultMessage: "content-type"
1122
- }),
1123
- name: "content-type"
1124
- }
1125
- ),
1126
- /* @__PURE__ */ jsx(
1127
- Table.HeaderCell,
1128
- {
1129
- attribute: { type: "string" },
1130
- label: formatMessage({
1131
- id: "content-releases.page.ReleaseDetails.table.header.label.action",
1132
- defaultMessage: "action"
1133
- }),
1134
- name: "action"
1135
- }
1136
- ),
1137
- !release.releasedAt && /* @__PURE__ */ jsx(
1138
- Table.HeaderCell,
1139
- {
1140
- attribute: { type: "string" },
1141
- label: formatMessage({
1142
- id: "content-releases.page.ReleaseDetails.table.header.label.status",
1143
- defaultMessage: "status"
1144
- }),
1145
- name: "status"
1146
- }
1147
- )
1148
- ] }),
1149
- /* @__PURE__ */ jsx(Table.LoadingBody, {}),
1150
- /* @__PURE__ */ jsx(Table.Body, { children: releaseActions[key].map(
1151
- ({ id, contentType, locale, type, entry }, actionIndex) => /* @__PURE__ */ jsxs(Tr, { children: [
1152
- /* @__PURE__ */ jsx(Td, { width: "25%", maxWidth: "200px", children: /* @__PURE__ */ jsx(Typography, { ellipsis: true, children: `${contentType.mainFieldValue || entry.id}` }) }),
1153
- /* @__PURE__ */ jsx(Td, { width: "10%", children: /* @__PURE__ */ jsx(Typography, { children: `${locale?.name ? locale.name : "-"}` }) }),
1154
- /* @__PURE__ */ jsx(Td, { width: "10%", children: /* @__PURE__ */ jsx(Typography, { children: contentType.displayName || "" }) }),
1155
- /* @__PURE__ */ jsx(Td, { width: "20%", children: release.releasedAt ? /* @__PURE__ */ jsx(Typography, { children: formatMessage(
1156
- {
1157
- id: "content-releases.page.ReleaseDetails.table.action-published",
1158
- defaultMessage: "This entry was <b>{isPublish, select, true {published} other {unpublished}}</b>."
1159
- },
1160
- {
1161
- isPublish: type === "publish",
1162
- b: (children) => /* @__PURE__ */ jsx(Typography, { fontWeight: "bold", children })
1163
- }
1164
- ) }) : /* @__PURE__ */ jsx(
1165
- ReleaseActionOptions,
1166
- {
1167
- selected: type,
1168
- handleChange: (e) => handleChangeType(e, id, [key, actionIndex]),
1169
- name: `release-action-${id}-type`,
1170
- disabled: !canUpdate
1171
- }
1172
- ) }),
1173
- !release.releasedAt && /* @__PURE__ */ jsxs(Fragment, { children: [
1174
- /* @__PURE__ */ jsx(Td, { width: "20%", minWidth: "200px", children: /* @__PURE__ */ jsx(
1175
- EntryValidationText,
1176
- {
1177
- action: type,
1178
- schema: contentTypes?.[contentType.uid],
1179
- components,
1180
- entry
1181
- }
1182
- ) }),
1183
- /* @__PURE__ */ jsx(Td, { children: /* @__PURE__ */ jsx(Flex, { justifyContent: "flex-end", children: /* @__PURE__ */ jsxs(ReleaseActionMenu.Root, { children: [
1184
- /* @__PURE__ */ jsx(
1185
- ReleaseActionMenu.ReleaseActionEntryLinkItem,
1186
- {
1187
- contentTypeUid: contentType.uid,
1188
- entryId: entry.id,
1189
- locale: locale?.code
1190
- }
1191
- ),
1192
- /* @__PURE__ */ jsx(
1193
- ReleaseActionMenu.DeleteReleaseActionItem,
1194
- {
1195
- releaseId: release.id,
1196
- actionId: id
1197
- }
1198
- )
1199
- ] }) }) })
1200
- ] })
1201
- ] }, id)
1202
- ) })
1203
- ] })
1204
- }
1205
- )
1206
- ] }, `releases-group-${key}`)),
1207
- /* @__PURE__ */ jsxs(
1208
- Pagination.Root,
1209
- {
1210
- ...releaseMeta?.pagination,
1211
- defaultPageSize: releaseMeta?.pagination?.pageSize,
1212
- children: [
1213
- /* @__PURE__ */ jsx(Pagination.PageSize, {}),
1214
- /* @__PURE__ */ jsx(Pagination.Links, {})
1215
- ]
1216
- }
1217
- )
1218
- ] }) });
1219
- };
1220
- const ReleaseDetailsPage = () => {
1221
- const { formatMessage } = useIntl();
1222
- const { releaseId } = useParams();
1223
- const toggleNotification = useNotification();
1224
- const { formatAPIError } = useAPIErrorHandler();
1225
- const navigate = useNavigate();
1226
- const [releaseModalShown, setReleaseModalShown] = React.useState(false);
1227
- const [showWarningSubmit, setWarningSubmit] = React.useState(false);
1228
- const {
1229
- isLoading: isLoadingDetails,
1230
- data,
1231
- isSuccess: isSuccessDetails
1232
- } = useGetReleaseQuery(
1233
- { id: releaseId },
1234
- {
1235
- skip: !releaseId
1236
- }
1237
- );
1238
- const [updateRelease, { isLoading: isSubmittingForm }] = useUpdateReleaseMutation();
1239
- const [deleteRelease, { isLoading: isDeletingRelease }] = useDeleteReleaseMutation();
1240
- const toggleEditReleaseModal = () => {
1241
- setReleaseModalShown((prev) => !prev);
1242
- };
1243
- const toggleWarningSubmit = () => setWarningSubmit((prevState) => !prevState);
1244
- if (isLoadingDetails) {
1245
- return /* @__PURE__ */ jsx(
1246
- ReleaseDetailsLayout,
1247
- {
1248
- toggleEditReleaseModal,
1249
- toggleWarningSubmit,
1250
- children: /* @__PURE__ */ jsx(Page.Loading, {})
1251
- }
1252
- );
1253
- }
1254
- if (!releaseId) {
1255
- return /* @__PURE__ */ jsx(Navigate, { to: ".." });
1256
- }
1257
- const releaseData = isSuccessDetails && data?.data || null;
1258
- const title = releaseData?.name || "";
1259
- const timezone = releaseData?.timezone ?? null;
1260
- const scheduledAt = releaseData?.scheduledAt && timezone ? utcToZonedTime(releaseData.scheduledAt, timezone) : null;
1261
- const date = scheduledAt ? new Date(format(scheduledAt, "yyyy-MM-dd")) : null;
1262
- const time = scheduledAt ? format(scheduledAt, "HH:mm") : "";
1263
- const handleEditRelease = async (values) => {
1264
- const response = await updateRelease({
1265
- id: releaseId,
1266
- name: values.name,
1267
- scheduledAt: values.scheduledAt,
1268
- timezone: values.timezone
1269
- });
1270
- if ("data" in response) {
1271
- toggleNotification({
1272
- type: "success",
1273
- message: formatMessage({
1274
- id: "content-releases.modal.release-updated-notification-success",
1275
- defaultMessage: "Release updated."
1276
- })
1277
- });
1278
- } else if (isAxiosError(response.error)) {
1279
- toggleNotification({
1280
- type: "warning",
1281
- message: formatAPIError(response.error)
1282
- });
1283
- } else {
1284
- toggleNotification({
1285
- type: "warning",
1286
- message: formatMessage({ id: "notification.error", defaultMessage: "An error occurred" })
1287
- });
1288
- }
1289
- toggleEditReleaseModal();
1290
- };
1291
- const handleDeleteRelease = async () => {
1292
- const response = await deleteRelease({
1293
- id: releaseId
1294
- });
1295
- if ("data" in response) {
1296
- navigate("..");
1297
- } else if (isAxiosError(response.error)) {
1298
- toggleNotification({
1299
- type: "warning",
1300
- message: formatAPIError(response.error)
1301
- });
1302
- } else {
1303
- toggleNotification({
1304
- type: "warning",
1305
- message: formatMessage({ id: "notification.error", defaultMessage: "An error occurred" })
1306
- });
1307
- }
1308
- };
1309
- return /* @__PURE__ */ jsxs(
1310
- ReleaseDetailsLayout,
1311
- {
1312
- toggleEditReleaseModal,
1313
- toggleWarningSubmit,
1314
- children: [
1315
- /* @__PURE__ */ jsx(ReleaseDetailsBody, { releaseId }),
1316
- releaseModalShown && /* @__PURE__ */ jsx(
1317
- ReleaseModal,
1318
- {
1319
- handleClose: toggleEditReleaseModal,
1320
- handleSubmit: handleEditRelease,
1321
- isLoading: isLoadingDetails || isSubmittingForm,
1322
- initialValues: {
1323
- name: title || "",
1324
- scheduledAt,
1325
- date,
1326
- time,
1327
- isScheduled: Boolean(scheduledAt),
1328
- timezone
1329
- }
1330
- }
1331
- ),
1332
- /* @__PURE__ */ jsx(
1333
- ConfirmDialog,
1334
- {
1335
- bodyText: {
1336
- id: "content-releases.dialog.confirmation-message",
1337
- defaultMessage: "Are you sure you want to delete this release?"
1338
- },
1339
- isOpen: showWarningSubmit,
1340
- isConfirmButtonLoading: isDeletingRelease,
1341
- onToggleDialog: toggleWarningSubmit,
1342
- onConfirm: handleDeleteRelease
1343
- }
1344
- )
1345
- ]
1346
- }
1347
- );
1348
- };
1349
- const App = () => {
1350
- return /* @__PURE__ */ jsx(Page.Protect, { permissions: PERMISSIONS.main, children: /* @__PURE__ */ jsxs(Routes, { children: [
1351
- /* @__PURE__ */ jsx(Route, { index: true, element: /* @__PURE__ */ jsx(ReleasesPage, {}) }),
1352
- /* @__PURE__ */ jsx(Route, { path: ":releaseId", element: /* @__PURE__ */ jsx(ReleaseDetailsPage, {}) })
1353
- ] }) });
1354
- };
1355
- export {
1356
- App
1357
- };
1358
- //# sourceMappingURL=App-0Er6xxcq.mjs.map