@strapi/content-releases 0.0.0-experimental.check-license → 0.0.0-experimental.d23c1d5b0e45dd06ef09977f526c85468be05403

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