@strapi/content-releases 0.0.0-next.e09d30edcbd16960a838997778a31d50e9c60bc4 → 0.0.0-next.e61eff51f9834ffdef16bdc236aecab5f837723b

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