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

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