@strapi/content-releases 0.0.0-next.4af8963f6880c5fb9fae32ecd580f5cd33eaddda → 0.0.0-next.4afb25b82502b04f05bbc122b43e1e3a806e7600

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