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

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