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