@strapi/i18n 0.0.0-experimental.a65a85fdea97faae8679d3ffc5f9d79af61abd26 → 0.0.0-experimental.a6728ad43ac70ae19dabb624dbfca1f2d9610a86

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 (53) hide show
  1. package/LICENSE +18 -3
  2. package/dist/_chunks/{SettingsPage-pvdzbKEM.mjs → SettingsPage-BIZrSFGY.mjs} +118 -129
  3. package/dist/_chunks/SettingsPage-BIZrSFGY.mjs.map +1 -0
  4. package/dist/_chunks/{SettingsPage-B73NO_8-.js → SettingsPage-kMDCxWLw.js} +117 -128
  5. package/dist/_chunks/SettingsPage-kMDCxWLw.js.map +1 -0
  6. package/dist/_chunks/{en-BuBc6LKZ.js → en-B6327hMz.js} +13 -2
  7. package/dist/_chunks/en-B6327hMz.js.map +1 -0
  8. package/dist/_chunks/{en-CnrTsjWS.mjs → en-DZXpOMHo.mjs} +13 -2
  9. package/dist/_chunks/en-DZXpOMHo.mjs.map +1 -0
  10. package/dist/_chunks/{index-lckTPHiZ.mjs → index-DXrgAtCA.mjs} +721 -208
  11. package/dist/_chunks/index-DXrgAtCA.mjs.map +1 -0
  12. package/dist/_chunks/{index-kcYwoCdE.js → index-Dncj9Inq.js} +722 -211
  13. package/dist/_chunks/index-Dncj9Inq.js.map +1 -0
  14. package/dist/admin/index.js +1 -1
  15. package/dist/admin/index.mjs +2 -2
  16. package/dist/admin/src/components/BulkLocaleActionModal.d.ts +16 -0
  17. package/dist/admin/src/components/CMHeaderActions.d.ts +33 -3
  18. package/dist/admin/src/components/EditLocale.d.ts +5 -4
  19. package/dist/admin/src/contentReleasesHooks/releaseDetailsView.d.ts +9 -5
  20. package/dist/admin/src/services/api.d.ts +2 -3
  21. package/dist/admin/src/services/locales.d.ts +1 -1
  22. package/dist/admin/src/services/relations.d.ts +7 -0
  23. package/dist/admin/src/utils/baseQuery.d.ts +4 -19
  24. package/dist/admin/src/utils/clean.d.ts +4 -0
  25. package/dist/admin/src/utils/schemas.d.ts +1 -0
  26. package/dist/server/index.js +61 -98
  27. package/dist/server/index.js.map +1 -1
  28. package/dist/server/index.mjs +62 -99
  29. package/dist/server/index.mjs.map +1 -1
  30. package/dist/server/src/bootstrap.d.ts +1 -4
  31. package/dist/server/src/bootstrap.d.ts.map +1 -1
  32. package/dist/server/src/index.d.ts +15 -13
  33. package/dist/server/src/index.d.ts.map +1 -1
  34. package/dist/server/src/services/index.d.ts +14 -10
  35. package/dist/server/src/services/index.d.ts.map +1 -1
  36. package/dist/server/src/services/permissions/actions.d.ts +14 -2
  37. package/dist/server/src/services/permissions/actions.d.ts.map +1 -1
  38. package/dist/server/src/services/permissions.d.ts +14 -2
  39. package/dist/server/src/services/permissions.d.ts.map +1 -1
  40. package/dist/server/src/utils/index.d.ts +0 -2
  41. package/dist/server/src/utils/index.d.ts.map +1 -1
  42. package/dist/shared/contracts/content-manager.d.ts +20 -1
  43. package/package.json +15 -16
  44. package/dist/_chunks/SettingsPage-B73NO_8-.js.map +0 -1
  45. package/dist/_chunks/SettingsPage-pvdzbKEM.mjs.map +0 -1
  46. package/dist/_chunks/en-BuBc6LKZ.js.map +0 -1
  47. package/dist/_chunks/en-CnrTsjWS.mjs.map +0 -1
  48. package/dist/_chunks/index-kcYwoCdE.js.map +0 -1
  49. package/dist/_chunks/index-lckTPHiZ.mjs.map +0 -1
  50. package/dist/admin/src/components/Initializer.d.ts +0 -5
  51. package/dist/server/src/services/entity-service-decorator.d.ts +0 -29
  52. package/dist/server/src/services/entity-service-decorator.d.ts.map +0 -1
  53. package/strapi-server.js +0 -3
@@ -6,12 +6,11 @@ const React = require("react");
6
6
  const designSystem = require("@strapi/design-system");
7
7
  const icons = require("@strapi/icons");
8
8
  const reactIntl = require("react-intl");
9
- const styled = require("styled-components");
9
+ const styledComponents = require("styled-components");
10
+ const query = require("@reduxjs/toolkit/query");
10
11
  const strapiAdmin = require("@strapi/admin/strapi-admin");
11
12
  const strapiAdmin$1 = require("@strapi/content-manager/strapi-admin");
12
13
  const reactRouterDom = require("react-router-dom");
13
- const react = require("@reduxjs/toolkit/query/react");
14
- const axios = require("axios");
15
14
  const qs = require("qs");
16
15
  const omit = require("lodash/omit");
17
16
  const _interopDefault = (e) => e && e.__esModule ? e : { default: e };
@@ -36,7 +35,6 @@ function _interopNamespace(e) {
36
35
  const get__default = /* @__PURE__ */ _interopDefault(get);
37
36
  const yup__namespace = /* @__PURE__ */ _interopNamespace(yup);
38
37
  const React__namespace = /* @__PURE__ */ _interopNamespace(React);
39
- const styled__default = /* @__PURE__ */ _interopDefault(styled);
40
38
  const qs__namespace = /* @__PURE__ */ _interopNamespace(qs);
41
39
  const omit__default = /* @__PURE__ */ _interopDefault(omit);
42
40
  const __variableDynamicImportRuntimeHelper = (glob, path) => {
@@ -50,7 +48,7 @@ const __variableDynamicImportRuntimeHelper = (glob, path) => {
50
48
  };
51
49
  const pluginId = "i18n";
52
50
  const getTranslation = (id) => `${pluginId}.${id}`;
53
- const TextAlignTypography = styled__default.default(designSystem.Typography)`
51
+ const TextAlignTypography = styledComponents.styled(designSystem.Typography)`
54
52
  text-align: center;
55
53
  `;
56
54
  const CheckboxConfirmation = ({
@@ -74,9 +72,7 @@ const CheckboxConfirmation = ({
74
72
  };
75
73
  const handleConfirm = () => {
76
74
  onChange({ target: { name, value: false, type: "checkbox" } });
77
- setIsOpen(false);
78
75
  };
79
- const handleToggle = () => setIsOpen((prev) => !prev);
80
76
  const label = intlLabel.id ? formatMessage(
81
77
  { id: intlLabel.id, defaultMessage: intlLabel.defaultMessage },
82
78
  { ...intlLabel.values }
@@ -85,43 +81,36 @@ const CheckboxConfirmation = ({
85
81
  { id: description.id, defaultMessage: description.defaultMessage },
86
82
  { ...description.values }
87
83
  ) : "";
88
- return /* @__PURE__ */ jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [
89
- /* @__PURE__ */ jsxRuntime.jsx(
90
- designSystem.Checkbox,
91
- {
92
- hint,
93
- id: name,
94
- name,
95
- onValueChange: handleChange,
96
- value,
97
- type: "checkbox",
98
- children: label
99
- }
100
- ),
101
- isOpen && /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Dialog, { onClose: handleToggle, title: "Confirmation", isOpen, children: [
102
- /* @__PURE__ */ jsxRuntime.jsx(designSystem.DialogBody, { icon: /* @__PURE__ */ jsxRuntime.jsx(icons.WarningCircle, {}), children: /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Flex, { direction: "column", alignItems: "stretch", gap: 2, children: [
103
- /* @__PURE__ */ jsxRuntime.jsx(designSystem.Flex, { justifyContent: "center", children: /* @__PURE__ */ jsxRuntime.jsx(TextAlignTypography, { id: "confirm-description", children: formatMessage({
84
+ return /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Dialog.Root, { open: isOpen, onOpenChange: setIsOpen, children: [
85
+ /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Field.Root, { hint, name, children: [
86
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Checkbox, { onCheckedChange: handleChange, checked: value, children: label }),
87
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Field.Hint, {})
88
+ ] }),
89
+ /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Dialog.Content, { children: [
90
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Dialog.Header, { children: formatMessage({
91
+ id: getTranslation("CheckboxConfirmation.Modal.title"),
92
+ defaultMessage: "Disable localization"
93
+ }) }),
94
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Dialog.Body, { icon: /* @__PURE__ */ jsxRuntime.jsx(icons.WarningCircle, {}), children: /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Flex, { direction: "column", alignItems: "stretch", gap: 2, children: [
95
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Flex, { justifyContent: "center", children: /* @__PURE__ */ jsxRuntime.jsx(TextAlignTypography, { children: formatMessage({
104
96
  id: getTranslation("CheckboxConfirmation.Modal.content"),
105
97
  defaultMessage: "Disabling localization will engender the deletion of all your content but the one associated to your default locale (if existing)."
106
98
  }) }) }),
107
- /* @__PURE__ */ jsxRuntime.jsx(designSystem.Flex, { justifyContent: "center", children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { fontWeight: "semiBold", id: "confirm-description", children: formatMessage({
99
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Flex, { justifyContent: "center", children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { fontWeight: "semiBold", children: formatMessage({
108
100
  id: getTranslation("CheckboxConfirmation.Modal.body"),
109
101
  defaultMessage: "Do you want to disable it?"
110
102
  }) }) })
111
103
  ] }) }),
112
- /* @__PURE__ */ jsxRuntime.jsx(
113
- designSystem.DialogFooter,
114
- {
115
- startAction: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Button, { onClick: handleToggle, variant: "tertiary", children: formatMessage({
116
- id: "components.popUpWarning.button.cancel",
117
- defaultMessage: "No, cancel"
118
- }) }),
119
- endAction: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Button, { variant: "danger-light", onClick: handleConfirm, children: formatMessage({
120
- id: getTranslation("CheckboxConfirmation.Modal.button-confirm"),
121
- defaultMessage: "Yes, disable"
122
- }) })
123
- }
124
- )
104
+ /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Dialog.Footer, { children: [
105
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Dialog.Cancel, { children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Button, { variant: "tertiary", children: formatMessage({
106
+ id: "components.popUpWarning.button.cancel",
107
+ defaultMessage: "No, cancel"
108
+ }) }) }),
109
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Dialog.Action, { children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Button, { variant: "danger-light", onClick: handleConfirm, children: formatMessage({
110
+ id: getTranslation("CheckboxConfirmation.Modal.button-confirm"),
111
+ defaultMessage: "Yes, disable"
112
+ }) }) })
113
+ ] })
125
114
  ] })
126
115
  ] });
127
116
  };
@@ -172,7 +161,7 @@ const useI18n = () => {
172
161
  model: params.slug
173
162
  },
174
163
  {
175
- skip: !params.slug || !params.collectionType
164
+ skip: true
176
165
  }
177
166
  );
178
167
  if (doesPluginOptionsHaveI18nLocalized(schema?.pluginOptions)) {
@@ -186,64 +175,8 @@ const useI18n = () => {
186
175
  ...actions
187
176
  };
188
177
  };
189
- const axiosBaseQuery = () => async (query, { signal }) => {
190
- try {
191
- const { get: get2, post, del, put } = strapiAdmin.getFetchClient();
192
- if (typeof query === "string") {
193
- const result = await get2(query, { signal });
194
- return { data: result.data };
195
- } else {
196
- const { url, method = "GET", data, config } = query;
197
- if (method === "POST") {
198
- const result2 = await post(url, data, { ...config, signal });
199
- return { data: result2.data };
200
- }
201
- if (method === "DELETE") {
202
- const result2 = await del(url, { ...config, signal });
203
- return { data: result2.data };
204
- }
205
- if (method === "PUT") {
206
- const result2 = await put(url, data, { ...config, signal });
207
- return { data: result2.data };
208
- }
209
- const result = await get2(url, { ...config, signal });
210
- return { data: result.data };
211
- }
212
- } catch (err) {
213
- if (axios.isAxiosError(err)) {
214
- if (typeof err.response?.data === "object" && err.response?.data !== null && "error" in err.response?.data) {
215
- return { data: void 0, error: err.response?.data.error };
216
- } else {
217
- return {
218
- data: void 0,
219
- error: {
220
- name: "UnknownError",
221
- message: "There was an unknown error response from the API",
222
- details: err.response?.data,
223
- status: err.response?.status
224
- }
225
- };
226
- }
227
- }
228
- const error = err;
229
- return {
230
- data: void 0,
231
- error: {
232
- name: error.name,
233
- message: error.message,
234
- stack: error.stack
235
- }
236
- };
237
- }
238
- };
239
- const isBaseQueryError = (error) => {
240
- return error.name !== void 0;
241
- };
242
- const i18nApi = react.createApi({
243
- reducerPath: "i18nApi",
244
- baseQuery: axiosBaseQuery(),
245
- tagTypes: ["Locale"],
246
- endpoints: () => ({})
178
+ const i18nApi = strapiAdmin.adminApi.enhanceEndpoints({
179
+ addTagTypes: ["Locale"]
247
180
  });
248
181
  const localesApi = i18nApi.injectEndpoints({
249
182
  endpoints: (builder) => ({
@@ -292,6 +225,295 @@ const {
292
225
  useGetDefaultLocalesQuery,
293
226
  useUpdateLocaleMutation
294
227
  } = localesApi;
228
+ const relationsApi = i18nApi.injectEndpoints({
229
+ overrideExisting: true,
230
+ endpoints: (builder) => ({
231
+ getManyDraftRelationCount: builder.query({
232
+ query: ({ model, ...params }) => ({
233
+ url: `/content-manager/collection-types/${model}/actions/countManyEntriesDraftRelations`,
234
+ method: "GET",
235
+ config: {
236
+ params
237
+ }
238
+ }),
239
+ transformResponse: (response) => response.data
240
+ })
241
+ })
242
+ });
243
+ const { useGetManyDraftRelationCountQuery } = relationsApi;
244
+ const cleanData = (data, schema, components) => {
245
+ const cleanedData = removeFields(data, [
246
+ "createdAt",
247
+ "createdBy",
248
+ "updatedAt",
249
+ "updatedBy",
250
+ "id",
251
+ "documentId",
252
+ "publishedAt",
253
+ "strapi_stage",
254
+ "strapi_assignee",
255
+ "locale"
256
+ ]);
257
+ const cleanedDataWithoutPasswordAndRelation = recursiveRemoveFieldTypes(
258
+ cleanedData,
259
+ schema,
260
+ components,
261
+ ["relation", "password"]
262
+ );
263
+ return cleanedDataWithoutPasswordAndRelation;
264
+ };
265
+ const removeFields = (data, fields) => {
266
+ return Object.keys(data).reduce((acc, current) => {
267
+ if (fields.includes(current)) {
268
+ return acc;
269
+ }
270
+ acc[current] = data[current];
271
+ return acc;
272
+ }, {});
273
+ };
274
+ const recursiveRemoveFieldTypes = (data, schema, components, fields) => {
275
+ return Object.keys(data).reduce((acc, current) => {
276
+ const attribute = schema.attributes[current] ?? { type: void 0 };
277
+ if (fields.includes(attribute.type)) {
278
+ return acc;
279
+ }
280
+ if (attribute.type === "dynamiczone") {
281
+ acc[current] = data[current].map((componentValue, index2) => {
282
+ const { id: _, ...rest } = recursiveRemoveFieldTypes(
283
+ componentValue,
284
+ components[componentValue.__component],
285
+ components,
286
+ fields
287
+ );
288
+ return {
289
+ ...rest,
290
+ __temp_key__: index2 + 1
291
+ };
292
+ });
293
+ } else if (attribute.type === "component") {
294
+ const { repeatable, component } = attribute;
295
+ if (repeatable) {
296
+ acc[current] = (data[current] ?? []).map((compoData, index2) => {
297
+ const { id: _, ...rest } = recursiveRemoveFieldTypes(
298
+ compoData,
299
+ components[component],
300
+ components,
301
+ fields
302
+ );
303
+ return {
304
+ ...rest,
305
+ __temp_key__: index2 + 1
306
+ };
307
+ });
308
+ } else {
309
+ const { id: _, ...rest } = recursiveRemoveFieldTypes(
310
+ data[current] ?? {},
311
+ components[component],
312
+ components,
313
+ fields
314
+ );
315
+ acc[current] = rest;
316
+ }
317
+ } else {
318
+ acc[current] = data[current];
319
+ }
320
+ return acc;
321
+ }, {});
322
+ };
323
+ const isErrorMessageDescriptor = (object) => {
324
+ return typeof object === "object" && object !== null && "id" in object && "defaultMessage" in object;
325
+ };
326
+ const EntryValidationText = ({
327
+ status = "draft",
328
+ validationErrors,
329
+ action
330
+ }) => {
331
+ const { formatMessage } = reactIntl.useIntl();
332
+ const getErrorStr = (key, value) => {
333
+ if (typeof value === "string") {
334
+ return `${key}: ${value}`;
335
+ } else if (isErrorMessageDescriptor(value)) {
336
+ return `${key}: ${formatMessage(value)}`;
337
+ } else if (Array.isArray(value)) {
338
+ return value.map((v) => getErrorStr(key, v)).join(" ");
339
+ } else if (typeof value === "object" && !Array.isArray(value)) {
340
+ return Object.entries(value).map(([k, v]) => getErrorStr(k, v)).join(" ");
341
+ } else {
342
+ return "";
343
+ }
344
+ };
345
+ if (validationErrors) {
346
+ const validationErrorsMessages = Object.entries(validationErrors).map(([key, value]) => {
347
+ return getErrorStr(key, value);
348
+ }).join(" ");
349
+ return /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Flex, { gap: 2, children: [
350
+ /* @__PURE__ */ jsxRuntime.jsx(icons.CrossCircle, { fill: "danger600" }),
351
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Tooltip, { label: validationErrorsMessages, children: /* @__PURE__ */ jsxRuntime.jsx(
352
+ designSystem.Typography,
353
+ {
354
+ maxWidth: "30rem",
355
+ textColor: "danger600",
356
+ variant: "omega",
357
+ fontWeight: "semiBold",
358
+ ellipsis: true,
359
+ children: validationErrorsMessages
360
+ }
361
+ ) })
362
+ ] });
363
+ }
364
+ const getStatusMessage = () => {
365
+ if (action === "bulk-publish") {
366
+ if (status === "published") {
367
+ return {
368
+ icon: /* @__PURE__ */ jsxRuntime.jsx(icons.CheckCircle, { fill: "success600" }),
369
+ text: formatMessage({
370
+ id: "content-manager.bulk-publish.already-published",
371
+ defaultMessage: "Already Published"
372
+ }),
373
+ textColor: "success600",
374
+ fontWeight: "bold"
375
+ };
376
+ } else if (status === "modified") {
377
+ return {
378
+ icon: /* @__PURE__ */ jsxRuntime.jsx(icons.ArrowsCounterClockwise, { fill: "alternative600" }),
379
+ text: formatMessage({
380
+ id: "app.utils.ready-to-publish-changes",
381
+ defaultMessage: "Ready to publish changes"
382
+ })
383
+ };
384
+ } else {
385
+ return {
386
+ icon: /* @__PURE__ */ jsxRuntime.jsx(icons.CheckCircle, { fill: "success600" }),
387
+ text: formatMessage({
388
+ id: "app.utils.ready-to-publish",
389
+ defaultMessage: "Ready to publish"
390
+ })
391
+ };
392
+ }
393
+ } else {
394
+ if (status === "draft") {
395
+ return {
396
+ icon: /* @__PURE__ */ jsxRuntime.jsx(icons.CheckCircle, { fill: "success600" }),
397
+ text: formatMessage({
398
+ id: "content-manager.bulk-unpublish.already-unpublished",
399
+ defaultMessage: "Already Unpublished"
400
+ }),
401
+ textColor: "success600",
402
+ fontWeight: "bold"
403
+ };
404
+ } else {
405
+ return {
406
+ icon: /* @__PURE__ */ jsxRuntime.jsx(icons.CheckCircle, { fill: "success600" }),
407
+ text: formatMessage({
408
+ id: "app.utils.ready-to-unpublish-changes",
409
+ defaultMessage: "Ready to unpublish"
410
+ }),
411
+ textColor: "success600",
412
+ fontWeight: "bold"
413
+ };
414
+ }
415
+ }
416
+ };
417
+ const { icon, text, textColor = "success600", fontWeight = "normal" } = getStatusMessage();
418
+ return /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Flex, { gap: 2, children: [
419
+ icon,
420
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { textColor, fontWeight, children: text })
421
+ ] });
422
+ };
423
+ const BoldChunk = (chunks) => /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { fontWeight: "bold", children: chunks });
424
+ const BulkLocaleActionModal = ({
425
+ headers,
426
+ rows,
427
+ localesMetadata,
428
+ validationErrors = {},
429
+ action
430
+ }) => {
431
+ const { formatMessage } = reactIntl.useIntl();
432
+ const selectedRows = strapiAdmin.useTable(
433
+ "BulkLocaleActionModal",
434
+ (state) => state.selectedRows
435
+ );
436
+ const getFormattedCountMessage = () => {
437
+ const currentStatusByLocale = rows.reduce((acc, { locale, status }) => {
438
+ acc[locale] = status;
439
+ return acc;
440
+ }, {});
441
+ const localesWithErrors = Object.keys(validationErrors);
442
+ const publishedCount = selectedRows.filter(
443
+ ({ locale }) => currentStatusByLocale[locale] === "published"
444
+ ).length;
445
+ const draftCount = selectedRows.filter(
446
+ ({ locale }) => (currentStatusByLocale[locale] === "draft" || currentStatusByLocale[locale] === "modified") && !localesWithErrors.includes(locale)
447
+ ).length;
448
+ const withErrorsCount = localesWithErrors.length;
449
+ const messageId = action === "bulk-publish" ? "content-manager.containers.list.selectedEntriesModal.selectedCount.publish" : "content-manager.containers.list.selectedEntriesModal.selectedCount.unpublish";
450
+ const defaultMessage = action === "bulk-publish" ? "<b>{publishedCount}</b> {publishedCount, plural, =0 {entries} one {entry} other {entries}} already published. <b>{draftCount}</b> {draftCount, plural, =0 {entries} one {entry} other {entries}} ready to publish. <b>{withErrorsCount}</b> {withErrorsCount, plural, =0 {entries} one {entry} other {entries}} waiting for action." : "<b>{draftCount}</b> {draftCount, plural, =0 {entries} one {entry} other {entries}} already unpublished. <b>{publishedCount}</b> {publishedCount, plural, =0 {entries} one {entry} other {entries}} ready to unpublish.";
451
+ return formatMessage(
452
+ {
453
+ id: messageId,
454
+ defaultMessage
455
+ },
456
+ {
457
+ withErrorsCount,
458
+ draftCount,
459
+ publishedCount,
460
+ b: BoldChunk
461
+ }
462
+ );
463
+ };
464
+ return /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Modal.Body, { children: [
465
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { children: getFormattedCountMessage() }),
466
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Box, { marginTop: 5, children: /* @__PURE__ */ jsxRuntime.jsxs(strapiAdmin.Table.Content, { children: [
467
+ /* @__PURE__ */ jsxRuntime.jsxs(strapiAdmin.Table.Head, { children: [
468
+ /* @__PURE__ */ jsxRuntime.jsx(strapiAdmin.Table.HeaderCheckboxCell, {}),
469
+ headers.map((head) => /* @__PURE__ */ jsxRuntime.jsx(strapiAdmin.Table.HeaderCell, { ...head }, head.name))
470
+ ] }),
471
+ /* @__PURE__ */ jsxRuntime.jsx(strapiAdmin.Table.Body, { children: rows.map(({ locale, status }, index2) => {
472
+ const error = validationErrors?.[locale] ?? null;
473
+ const statusVariant = status === "draft" ? "primary" : status === "published" ? "success" : "alternative";
474
+ return /* @__PURE__ */ jsxRuntime.jsxs(strapiAdmin.Table.Row, { children: [
475
+ /* @__PURE__ */ jsxRuntime.jsx(strapiAdmin.Table.CheckboxCell, { id: locale, "aria-label": `Select ${locale}` }),
476
+ /* @__PURE__ */ jsxRuntime.jsx(strapiAdmin.Table.Cell, { children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { variant: "sigma", textColor: "neutral600", children: Array.isArray(localesMetadata) ? localesMetadata.find((localeEntry) => localeEntry.code === locale)?.name : locale }) }),
477
+ /* @__PURE__ */ jsxRuntime.jsx(strapiAdmin.Table.Cell, { children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Box, { display: "flex", children: /* @__PURE__ */ jsxRuntime.jsx(
478
+ designSystem.Status,
479
+ {
480
+ display: "flex",
481
+ paddingLeft: "6px",
482
+ paddingRight: "6px",
483
+ paddingTop: "2px",
484
+ paddingBottom: "2px",
485
+ showBullet: false,
486
+ size: "S",
487
+ variant: statusVariant,
488
+ children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { tag: "span", variant: "pi", fontWeight: "bold", children: capitalize(status) })
489
+ }
490
+ ) }) }),
491
+ /* @__PURE__ */ jsxRuntime.jsx(strapiAdmin.Table.Cell, { children: /* @__PURE__ */ jsxRuntime.jsx(EntryValidationText, { validationErrors: error, status, action }) }),
492
+ /* @__PURE__ */ jsxRuntime.jsx(strapiAdmin.Table.Cell, { children: /* @__PURE__ */ jsxRuntime.jsx(
493
+ designSystem.IconButton,
494
+ {
495
+ tag: reactRouterDom.Link,
496
+ to: {
497
+ search: qs.stringify({ plugins: { i18n: { locale } } })
498
+ },
499
+ label: formatMessage(
500
+ {
501
+ id: getTranslation("Settings.list.actions.edit"),
502
+ defaultMessage: "Edit {name} locale"
503
+ },
504
+ {
505
+ name: locale
506
+ }
507
+ ),
508
+ variant: "ghost",
509
+ children: /* @__PURE__ */ jsxRuntime.jsx(icons.Pencil, {})
510
+ }
511
+ ) })
512
+ ] }, index2);
513
+ }) })
514
+ ] }) })
515
+ ] });
516
+ };
295
517
  const LocalePickerAction = ({
296
518
  document,
297
519
  meta,
@@ -300,42 +522,47 @@ const LocalePickerAction = ({
300
522
  documentId
301
523
  }) => {
302
524
  const { formatMessage } = reactIntl.useIntl();
303
- const [{ query }, setQuery] = strapiAdmin.useQueryParams();
525
+ const [{ query: query2 }, setQuery] = strapiAdmin.useQueryParams();
304
526
  const { hasI18n, canCreate, canRead } = useI18n();
305
527
  const { data: locales = [] } = useGetLocalesQuery();
306
- const { schema } = strapiAdmin$1.unstable_useDocument({ model, collectionType, documentId });
528
+ const currentDesiredLocale = query2.plugins?.i18n?.locale;
529
+ const { schema } = strapiAdmin$1.unstable_useDocument({
530
+ model,
531
+ collectionType,
532
+ documentId,
533
+ params: { locale: currentDesiredLocale }
534
+ });
307
535
  const handleSelect = React__namespace.useCallback(
308
536
  (value) => {
309
537
  setQuery({
310
538
  plugins: {
311
- ...query.plugins,
539
+ ...query2.plugins,
312
540
  i18n: {
313
541
  locale: value
314
542
  }
315
543
  }
316
544
  });
317
545
  },
318
- [query.plugins, setQuery]
546
+ [query2.plugins, setQuery]
319
547
  );
320
548
  React__namespace.useEffect(() => {
321
549
  if (!Array.isArray(locales) || !hasI18n) {
322
550
  return;
323
551
  }
324
- const currentDesiredLocale = query.plugins?.i18n?.locale;
325
552
  const doesLocaleExist = locales.find((loc) => loc.code === currentDesiredLocale);
326
553
  const defaultLocale = locales.find((locale) => locale.isDefault);
327
554
  if (!doesLocaleExist && defaultLocale?.code) {
328
555
  handleSelect(defaultLocale.code);
329
556
  }
330
- }, [handleSelect, hasI18n, locales, query.plugins?.i18n?.locale]);
331
- if (!hasI18n || !Array.isArray(locales) || locales.length === 0) {
332
- return null;
333
- }
334
- const currentLocale = query.plugins?.i18n?.locale || locales.find((loc) => loc.isDefault)?.code;
557
+ }, [handleSelect, hasI18n, locales, currentDesiredLocale]);
558
+ const currentLocale = Array.isArray(locales) ? locales.find((locale) => locale.code === currentDesiredLocale)?.code : void 0;
335
559
  const allCurrentLocales = [
336
560
  { status: getDocumentStatus(document, meta), locale: currentLocale },
337
561
  ...meta?.availableLocales ?? []
338
562
  ];
563
+ if (!hasI18n || !Array.isArray(locales) || locales.length === 0) {
564
+ return null;
565
+ }
339
566
  return {
340
567
  label: formatMessage({
341
568
  id: getTranslation("Settings.locales.modal.locales.label"),
@@ -347,7 +574,7 @@ const LocalePickerAction = ({
347
574
  );
348
575
  const status = currentLocaleDoc?.status ?? "draft";
349
576
  const permissionsToCheck = currentLocaleDoc ? canCreate : canRead;
350
- const statusVariant = status === "draft" ? "primary" : status === "published" ? "success" : "alternative";
577
+ const statusVariant = status === "draft" ? "secondary" : status === "published" ? "success" : "alternative";
351
578
  return {
352
579
  disabled: !permissionsToCheck.includes(locale.code),
353
580
  value: locale.code,
@@ -363,7 +590,7 @@ const LocalePickerAction = ({
363
590
  showBullet: false,
364
591
  size: "S",
365
592
  variant: statusVariant,
366
- children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { as: "span", variant: "pi", fontWeight: "bold", children: capitalize(status) })
593
+ children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { tag: "span", variant: "pi", fontWeight: "bold", children: capitalize(status) })
367
594
  }
368
595
  ) : null
369
596
  };
@@ -383,6 +610,95 @@ const getDocumentStatus = (document, meta) => {
383
610
  }
384
611
  return docStatus;
385
612
  };
613
+ const FillFromAnotherLocaleAction = ({
614
+ documentId,
615
+ meta,
616
+ model,
617
+ collectionType
618
+ }) => {
619
+ const { formatMessage } = reactIntl.useIntl();
620
+ const [{ query: query2 }] = strapiAdmin.useQueryParams();
621
+ const currentDesiredLocale = query2.plugins?.i18n?.locale;
622
+ const [localeSelected, setLocaleSelected] = React__namespace.useState(null);
623
+ const setValues = strapiAdmin.useForm("FillFromAnotherLocale", (state) => state.setValues);
624
+ const { getDocument } = strapiAdmin$1.unstable_useDocumentActions();
625
+ const { schema, components } = strapiAdmin$1.unstable_useDocument({
626
+ model,
627
+ documentId,
628
+ collectionType,
629
+ params: { locale: currentDesiredLocale }
630
+ });
631
+ const { data: locales = [] } = useGetLocalesQuery();
632
+ const availableLocales = Array.isArray(locales) ? locales.filter((locale) => meta?.availableLocales.some((l) => l.locale === locale.code)) : [];
633
+ const fillFromLocale = (onClose) => async () => {
634
+ const response = await getDocument({
635
+ collectionType,
636
+ model,
637
+ documentId,
638
+ params: { locale: localeSelected }
639
+ });
640
+ if (!response || !schema) {
641
+ return;
642
+ }
643
+ const { data } = response;
644
+ const cleanedData = cleanData(data, schema, components);
645
+ setValues(cleanedData);
646
+ onClose();
647
+ };
648
+ return {
649
+ type: "icon",
650
+ icon: /* @__PURE__ */ jsxRuntime.jsx(icons.Download, {}),
651
+ disabled: availableLocales.length === 0,
652
+ label: formatMessage({
653
+ id: getTranslation("CMEditViewCopyLocale.copy-text"),
654
+ defaultMessage: "Fill in from another locale"
655
+ }),
656
+ dialog: {
657
+ type: "dialog",
658
+ title: formatMessage({
659
+ id: getTranslation("CMEditViewCopyLocale.dialog.title"),
660
+ defaultMessage: "Confirmation"
661
+ }),
662
+ content: ({ onClose }) => /* @__PURE__ */ jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [
663
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Dialog.Body, { children: /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Flex, { direction: "column", gap: 3, children: [
664
+ /* @__PURE__ */ jsxRuntime.jsx(icons.WarningCircle, { width: "24px", height: "24px", fill: "danger600" }),
665
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { textAlign: "center", children: formatMessage({
666
+ id: getTranslation("CMEditViewCopyLocale.dialog.body"),
667
+ defaultMessage: "Your current content will be erased and filled by the content of the selected locale:"
668
+ }) }),
669
+ /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Field.Root, { width: "100%", children: [
670
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Field.Label, { children: formatMessage({
671
+ id: getTranslation("CMEditViewCopyLocale.dialog.field.label"),
672
+ defaultMessage: "Locale"
673
+ }) }),
674
+ /* @__PURE__ */ jsxRuntime.jsx(
675
+ designSystem.SingleSelect,
676
+ {
677
+ value: localeSelected,
678
+ placeholder: formatMessage({
679
+ id: getTranslation("CMEditViewCopyLocale.dialog.field.placeholder"),
680
+ defaultMessage: "Select one locale..."
681
+ }),
682
+ onChange: (value) => setLocaleSelected(value),
683
+ children: availableLocales.map((locale) => /* @__PURE__ */ jsxRuntime.jsx(designSystem.SingleSelectOption, { value: locale.code, children: locale.name }, locale.code))
684
+ }
685
+ )
686
+ ] })
687
+ ] }) }),
688
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Dialog.Footer, { children: /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Flex, { gap: 2, width: "100%", children: [
689
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Button, { flex: "auto", variant: "tertiary", onClick: onClose, children: formatMessage({
690
+ id: getTranslation("CMEditViewCopyLocale.cancel-text"),
691
+ defaultMessage: "No, cancel"
692
+ }) }),
693
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Button, { flex: "auto", variant: "success", onClick: fillFromLocale(onClose), children: formatMessage({
694
+ id: getTranslation("CMEditViewCopyLocale.submit-text"),
695
+ defaultMessage: "Yes, fill in"
696
+ }) })
697
+ ] }) })
698
+ ] })
699
+ }
700
+ };
701
+ };
386
702
  const DeleteLocaleAction = ({
387
703
  document,
388
704
  documentId,
@@ -394,16 +710,23 @@ const DeleteLocaleAction = ({
394
710
  const { toggleNotification } = strapiAdmin.useNotification();
395
711
  const { delete: deleteAction } = strapiAdmin$1.unstable_useDocumentActions();
396
712
  const { hasI18n, canDelete } = useI18n();
713
+ const [{ query: query2 }] = strapiAdmin.useQueryParams();
714
+ const { data: locales = [] } = useGetLocalesQuery();
715
+ const currentDesiredLocale = query2.plugins?.i18n?.locale;
716
+ const locale = !("error" in locales) && locales.find((loc) => loc.code === currentDesiredLocale);
397
717
  if (!hasI18n) {
398
718
  return null;
399
719
  }
400
720
  return {
401
721
  disabled: document?.locale && !canDelete.includes(document.locale) || !document || !document.id,
402
722
  position: ["header", "table-row"],
403
- label: formatMessage({
404
- id: getTranslation("actions.delete.label"),
405
- defaultMessage: "Delete locale"
406
- }),
723
+ label: formatMessage(
724
+ {
725
+ id: getTranslation("actions.delete.label"),
726
+ defaultMessage: "Delete entry ({locale})"
727
+ },
728
+ { locale: locale && locale.name }
729
+ ),
407
730
  icon: /* @__PURE__ */ jsxRuntime.jsx(StyledTrash, {}),
408
731
  variant: "danger",
409
732
  dialog: {
@@ -414,7 +737,7 @@ const DeleteLocaleAction = ({
414
737
  }),
415
738
  content: /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Flex, { direction: "column", gap: 2, children: [
416
739
  /* @__PURE__ */ jsxRuntime.jsx(icons.WarningCircle, { width: "24px", height: "24px", fill: "danger600" }),
417
- /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { as: "p", variant: "omega", textAlign: "center", children: formatMessage({
740
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { tag: "p", variant: "omega", textAlign: "center", children: formatMessage({
418
741
  id: getTranslation("actions.delete.dialog.body"),
419
742
  defaultMessage: "Are you sure?"
420
743
  }) })
@@ -446,7 +769,263 @@ const DeleteLocaleAction = ({
446
769
  }
447
770
  };
448
771
  };
449
- const StyledTrash = styled__default.default(icons.Trash)`
772
+ const BulkLocaleAction = ({
773
+ document: baseDocument,
774
+ documentId,
775
+ model,
776
+ collectionType,
777
+ action
778
+ }) => {
779
+ const baseLocale = baseDocument?.locale ?? null;
780
+ const [{ query: query$1 }] = strapiAdmin.useQueryParams();
781
+ const params = React__namespace.useMemo(() => strapiAdmin$1.buildValidParams(query$1), [query$1]);
782
+ const isOnPublishedTab = query$1.status === "published";
783
+ const { formatMessage } = reactIntl.useIntl();
784
+ const { hasI18n, canPublish } = useI18n();
785
+ const { toggleNotification } = strapiAdmin.useNotification();
786
+ const { _unstableFormatAPIError: formatAPIError } = strapiAdmin.useAPIErrorHandler();
787
+ const [selectedRows, setSelectedRows] = React__namespace.useState([]);
788
+ const [isDraftRelationConfirmationOpen, setIsDraftRelationConfirmationOpen] = React__namespace.useState(false);
789
+ const { publishMany: publishManyAction, unpublishMany: unpublishManyAction } = strapiAdmin$1.unstable_useDocumentActions();
790
+ const {
791
+ document,
792
+ meta: documentMeta,
793
+ schema,
794
+ validate
795
+ } = strapiAdmin$1.unstable_useDocument(
796
+ {
797
+ model,
798
+ collectionType,
799
+ documentId,
800
+ params: {
801
+ locale: baseLocale
802
+ }
803
+ },
804
+ {
805
+ skip: !hasI18n || !baseLocale
806
+ }
807
+ );
808
+ const { data: localesMetadata = [] } = useGetLocalesQuery(hasI18n ? void 0 : query.skipToken);
809
+ const headers = [
810
+ {
811
+ label: formatMessage({
812
+ id: "global.name",
813
+ defaultMessage: "Name"
814
+ }),
815
+ name: "name"
816
+ },
817
+ {
818
+ label: formatMessage({
819
+ id: getTranslation("CMEditViewBulkLocale.status"),
820
+ defaultMessage: "Status"
821
+ }),
822
+ name: "status"
823
+ },
824
+ {
825
+ label: formatMessage({
826
+ id: getTranslation("CMEditViewBulkLocale.publication-status"),
827
+ defaultMessage: "Publication Status"
828
+ }),
829
+ name: "publication-status"
830
+ }
831
+ ];
832
+ const [rows, validationErrors] = React__namespace.useMemo(() => {
833
+ if (!document || !documentMeta?.availableLocales) {
834
+ return [[], {}];
835
+ }
836
+ const rowsFromMeta = documentMeta?.availableLocales.map((doc) => {
837
+ const { locale, status } = doc;
838
+ return { locale, status };
839
+ });
840
+ rowsFromMeta.unshift({
841
+ locale: document.locale,
842
+ status: document.status
843
+ });
844
+ const allDocuments = [document, ...documentMeta?.availableLocales ?? []];
845
+ const errors = allDocuments.reduce((errs, document2) => {
846
+ if (!document2) {
847
+ return errs;
848
+ }
849
+ const validation = validate(document2);
850
+ if (validation !== null) {
851
+ errs[document2.locale] = validation;
852
+ }
853
+ return errs;
854
+ }, {});
855
+ return [rowsFromMeta, errors];
856
+ }, [document, documentMeta?.availableLocales, validate]);
857
+ const isBulkPublish = action === "bulk-publish";
858
+ const localesForAction = selectedRows.reduce((acc, selectedRow) => {
859
+ const isValidLocale = (
860
+ // Validation errors are irrelevant if we are trying to unpublish
861
+ !isBulkPublish || !Object.keys(validationErrors).includes(selectedRow.locale)
862
+ );
863
+ const shouldAddLocale = isBulkPublish ? selectedRow.status !== "published" && isValidLocale : selectedRow.status !== "draft" && isValidLocale;
864
+ if (shouldAddLocale) {
865
+ acc.push(selectedRow.locale);
866
+ }
867
+ return acc;
868
+ }, []);
869
+ const enableDraftRelationsCount = false;
870
+ const {
871
+ data: draftRelationsCount = 0,
872
+ isLoading: isDraftRelationsLoading,
873
+ error: isDraftRelationsError
874
+ } = useGetManyDraftRelationCountQuery(
875
+ {
876
+ model,
877
+ documentIds: [documentId],
878
+ locale: localesForAction
879
+ },
880
+ {
881
+ skip: !enableDraftRelationsCount
882
+ }
883
+ );
884
+ React__namespace.useEffect(() => {
885
+ if (isDraftRelationsError) {
886
+ toggleNotification({
887
+ type: "danger",
888
+ message: formatAPIError(isDraftRelationsError)
889
+ });
890
+ }
891
+ }, [isDraftRelationsError, toggleNotification, formatAPIError]);
892
+ if (!schema?.options?.draftAndPublish) {
893
+ return null;
894
+ }
895
+ if (!hasI18n) {
896
+ return null;
897
+ }
898
+ if (!documentId) {
899
+ return null;
900
+ }
901
+ const publish = async () => {
902
+ await publishManyAction({
903
+ model,
904
+ documentIds: [documentId],
905
+ params: {
906
+ ...params,
907
+ locale: localesForAction
908
+ }
909
+ });
910
+ setSelectedRows([]);
911
+ };
912
+ const unpublish = async () => {
913
+ await unpublishManyAction({
914
+ model,
915
+ documentIds: [documentId],
916
+ params: {
917
+ ...params,
918
+ locale: localesForAction
919
+ }
920
+ });
921
+ setSelectedRows([]);
922
+ };
923
+ const handleAction = async () => {
924
+ if (draftRelationsCount > 0) {
925
+ setIsDraftRelationConfirmationOpen(true);
926
+ } else if (isBulkPublish) {
927
+ await publish();
928
+ } else {
929
+ await unpublish();
930
+ }
931
+ };
932
+ if (isDraftRelationConfirmationOpen) {
933
+ return {
934
+ label: formatMessage({
935
+ id: "app.components.ConfirmDialog.title",
936
+ defaultMessage: "Confirmation"
937
+ }),
938
+ variant: "danger",
939
+ dialog: {
940
+ onCancel: () => {
941
+ setIsDraftRelationConfirmationOpen(false);
942
+ },
943
+ onConfirm: async () => {
944
+ await publish();
945
+ setIsDraftRelationConfirmationOpen(false);
946
+ },
947
+ type: "dialog",
948
+ title: formatMessage({
949
+ id: getTranslation("actions.publish.dialog.title"),
950
+ defaultMessage: "Confirmation"
951
+ }),
952
+ content: /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Flex, { direction: "column", alignItems: "center", gap: 2, children: [
953
+ /* @__PURE__ */ jsxRuntime.jsx(icons.WarningCircle, { width: "2.4rem", height: "2.4rem", fill: "danger600" }),
954
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { textAlign: "center", children: formatMessage({
955
+ id: getTranslation("CMEditViewBulkLocale.draft-relation-warning"),
956
+ defaultMessage: "Some locales are related to draft entries. Publishing them could leave broken links in your app."
957
+ }) }),
958
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { textAlign: "center", children: formatMessage({
959
+ id: getTranslation("CMEditViewBulkLocale.continue-confirmation"),
960
+ defaultMessage: "Are you sure you want to continue?"
961
+ }) })
962
+ ] })
963
+ }
964
+ };
965
+ }
966
+ const hasPermission = selectedRows.map(({ locale }) => locale).every((locale) => canPublish.includes(locale));
967
+ return {
968
+ label: formatMessage({
969
+ id: getTranslation(`CMEditViewBulkLocale.${isBulkPublish ? "publish" : "unpublish"}-title`),
970
+ defaultMessage: `${isBulkPublish ? "Publish" : "Unpublish"} Multiple Locales`
971
+ }),
972
+ variant: isBulkPublish ? "secondary" : "danger",
973
+ icon: isBulkPublish ? /* @__PURE__ */ jsxRuntime.jsx(icons.ListPlus, {}) : /* @__PURE__ */ jsxRuntime.jsx(icons.Cross, {}),
974
+ disabled: isOnPublishedTab || canPublish.length === 0,
975
+ position: ["panel"],
976
+ dialog: {
977
+ type: "modal",
978
+ title: formatMessage({
979
+ id: getTranslation(`CMEditViewBulkLocale.${isBulkPublish ? "publish" : "unpublish"}-title`),
980
+ defaultMessage: `${isBulkPublish ? "Publish" : "Unpublish"} Multiple Locales`
981
+ }),
982
+ content: () => {
983
+ return /* @__PURE__ */ jsxRuntime.jsx(
984
+ strapiAdmin.Table.Root,
985
+ {
986
+ headers,
987
+ rows: rows.map((row) => ({
988
+ ...row,
989
+ id: row.locale
990
+ })),
991
+ selectedRows,
992
+ onSelectedRowsChange: (tableSelectedRows) => setSelectedRows(tableSelectedRows),
993
+ children: /* @__PURE__ */ jsxRuntime.jsx(
994
+ BulkLocaleActionModal,
995
+ {
996
+ validationErrors,
997
+ headers,
998
+ rows,
999
+ localesMetadata,
1000
+ action: action ?? "bulk-publish"
1001
+ }
1002
+ )
1003
+ }
1004
+ );
1005
+ },
1006
+ footer: () => /* @__PURE__ */ jsxRuntime.jsx(designSystem.Modal.Footer, { justifyContent: "flex-end", children: /* @__PURE__ */ jsxRuntime.jsx(
1007
+ designSystem.Button,
1008
+ {
1009
+ loading: isDraftRelationsLoading,
1010
+ disabled: !hasPermission || localesForAction.length === 0,
1011
+ variant: "default",
1012
+ onClick: handleAction,
1013
+ children: formatMessage({
1014
+ id: isBulkPublish ? "app.utils.publish" : "app.utils.unpublish",
1015
+ defaultMessage: isBulkPublish ? "Publish" : "Unpublish"
1016
+ })
1017
+ }
1018
+ ) })
1019
+ }
1020
+ };
1021
+ };
1022
+ const BulkLocalePublishAction = (props) => {
1023
+ return BulkLocaleAction({ action: "bulk-publish", ...props });
1024
+ };
1025
+ const BulkLocaleUnpublishAction = (props) => {
1026
+ return BulkLocaleAction({ action: "bulk-unpublish", ...props });
1027
+ };
1028
+ const StyledTrash = styledComponents.styled(icons.Trash)`
450
1029
  path {
451
1030
  fill: currentColor;
452
1031
  }
@@ -502,16 +1081,9 @@ const UnpublishModalAdditionalInfo = () => {
502
1081
  }
503
1082
  ) });
504
1083
  };
505
- const Initializer = ({ setPlugin }) => {
506
- const setPluginRef = React__namespace.useRef(setPlugin);
507
- React__namespace.useEffect(() => {
508
- setPluginRef.current(pluginId);
509
- }, []);
510
- return null;
511
- };
512
1084
  const LocalePicker = () => {
513
1085
  const { formatMessage } = reactIntl.useIntl();
514
- const [{ query }, setQuery] = strapiAdmin.useQueryParams();
1086
+ const [{ query: query2 }, setQuery] = strapiAdmin.useQueryParams();
515
1087
  const { hasI18n, canRead, canCreate } = useI18n();
516
1088
  const { data: locales = [] } = useGetLocalesQuery(void 0, {
517
1089
  skip: !hasI18n
@@ -521,25 +1093,25 @@ const LocalePicker = () => {
521
1093
  setQuery(
522
1094
  {
523
1095
  page: 1,
524
- plugins: { ...query.plugins, i18n: { locale: code } }
1096
+ plugins: { ...query2.plugins, i18n: { locale: code } }
525
1097
  },
526
1098
  "push",
527
1099
  replace
528
1100
  );
529
1101
  },
530
- [query.plugins, setQuery]
1102
+ [query2.plugins, setQuery]
531
1103
  );
532
1104
  React__namespace.useEffect(() => {
533
1105
  if (!Array.isArray(locales) || !hasI18n) {
534
1106
  return;
535
1107
  }
536
- const currentDesiredLocale = query.plugins?.i18n?.locale;
1108
+ const currentDesiredLocale = query2.plugins?.i18n?.locale;
537
1109
  const doesLocaleExist = locales.find((loc) => loc.code === currentDesiredLocale);
538
1110
  const defaultLocale = locales.find((locale) => locale.isDefault);
539
1111
  if (!doesLocaleExist && defaultLocale?.code) {
540
1112
  handleChange(defaultLocale.code, true);
541
1113
  }
542
- }, [hasI18n, handleChange, locales, query.plugins?.i18n?.locale]);
1114
+ }, [hasI18n, handleChange, locales, query2.plugins?.i18n?.locale]);
543
1115
  if (!hasI18n || !Array.isArray(locales) || locales.length === 0) {
544
1116
  return null;
545
1117
  }
@@ -554,7 +1126,7 @@ const LocalePicker = () => {
554
1126
  id: getTranslation("actions.select-locale"),
555
1127
  defaultMessage: "Select locale"
556
1128
  }),
557
- value: query.plugins?.i18n?.locale || locales.find((locale) => locale.isDefault)?.code,
1129
+ value: query2.plugins?.i18n?.locale || locales.find((locale) => locale.isDefault)?.code,
558
1130
  onChange: handleChange,
559
1131
  children: displayedLocales.map((locale) => /* @__PURE__ */ jsxRuntime.jsx(designSystem.SingleSelectOption, { value: locale.code, children: locale.name }, locale.id))
560
1132
  }
@@ -568,7 +1140,7 @@ const PERMISSIONS = {
568
1140
  read: [{ action: "plugin::i18n.locale.read", subject: null }]
569
1141
  };
570
1142
  const mutateEditViewHook = ({ layout }) => {
571
- if ("i18n" in layout.options && typeof layout.options.i18n === "object" && layout.options.i18n !== null && "localized" in layout.options.i18n && !layout.options.i18n.localized) {
1143
+ if (!("i18n" in layout.options) || typeof layout.options.i18n === "object" && layout.options.i18n !== null && "localized" in layout.options.i18n && !layout.options.i18n.localized) {
572
1144
  return { layout };
573
1145
  }
574
1146
  const components = Object.entries(layout.components).reduce(
@@ -613,8 +1185,8 @@ const doesFieldHaveI18nPluginOpt = (pluginOpts) => {
613
1185
  };
614
1186
  const LabelAction = ({ title, icon }) => {
615
1187
  const { formatMessage } = reactIntl.useIntl();
616
- return /* @__PURE__ */ jsxRuntime.jsxs(Span, { as: "span", children: [
617
- /* @__PURE__ */ jsxRuntime.jsx(designSystem.VisuallyHidden, { as: "span", children: `(${formatMessage(title)})` }),
1188
+ return /* @__PURE__ */ jsxRuntime.jsxs(Span, { tag: "span", children: [
1189
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.VisuallyHidden, { tag: "span", children: formatMessage(title) }),
618
1190
  React__namespace.cloneElement(icon, {
619
1191
  "aria-hidden": true,
620
1192
  focusable: false
@@ -622,7 +1194,7 @@ const LabelAction = ({ title, icon }) => {
622
1194
  })
623
1195
  ] });
624
1196
  };
625
- const Span = styled__default.default(designSystem.Flex)`
1197
+ const Span = styledComponents.styled(designSystem.Flex)`
626
1198
  svg {
627
1199
  width: 12px;
628
1200
  height: 12px;
@@ -649,13 +1221,7 @@ const LocaleListCell = ({
649
1221
  }
650
1222
  });
651
1223
  const { locale: language } = reactIntl.useIntl();
652
- const [visible, setVisible] = React__namespace.useState(false);
653
- const buttonRef = React__namespace.useRef(null);
654
1224
  const { data: locales = [] } = useGetLocalesQuery();
655
- const handleTogglePopover = (e) => {
656
- e.stopPropagation();
657
- setVisible((prev) => !prev);
658
- };
659
1225
  const formatter = designSystem.useCollator(language, {
660
1226
  sensitivity: "base"
661
1227
  });
@@ -677,64 +1243,14 @@ const LocaleListCell = ({
677
1243
  }
678
1244
  return locale.name;
679
1245
  }).toSorted((a, b) => formatter.compare(a, b));
680
- return /* @__PURE__ */ jsxRuntime.jsxs(Button, { type: "button", onClick: handleTogglePopover, ref: buttonRef, children: [
681
- /* @__PURE__ */ jsxRuntime.jsxs(
682
- ActionWrapper,
683
- {
684
- minWidth: "100%",
685
- alignItems: "center",
686
- justifyContent: "center",
687
- height: "3.2rem",
688
- width: "3.2rem",
689
- children: [
690
- /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { textColor: "neutral800", ellipsis: true, children: localesForDocument.join(", ") }),
691
- /* @__PURE__ */ jsxRuntime.jsx(designSystem.Flex, { children: /* @__PURE__ */ jsxRuntime.jsx(icons.CaretDown, {}) })
692
- ]
693
- }
694
- ),
695
- visible && /* @__PURE__ */ jsxRuntime.jsx(
696
- designSystem.Popover,
697
- {
698
- onDismiss: () => setVisible(false),
699
- source: buttonRef,
700
- spacing: 16,
701
- centered: true,
702
- children: /* @__PURE__ */ jsxRuntime.jsx("ul", { children: localesForDocument.map((name) => /* @__PURE__ */ jsxRuntime.jsx(designSystem.Box, { padding: 3, as: "li", children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { children: name }) }, name)) })
703
- }
704
- )
1246
+ return /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Popover.Root, { children: [
1247
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Popover.Trigger, { children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Button, { variant: "ghost", type: "button", onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Flex, { minWidth: "100%", alignItems: "center", justifyContent: "center", fontWeight: "regular", children: [
1248
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { textColor: "neutral800", ellipsis: true, marginRight: 2, children: localesForDocument.join(", ") }),
1249
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Flex, { children: /* @__PURE__ */ jsxRuntime.jsx(icons.CaretDown, { width: "1.2rem", height: "1.2rem" }) })
1250
+ ] }) }) }),
1251
+ /* @__PURE__ */ jsxRuntime.jsx(designSystem.Popover.Content, { sideOffset: 16, children: /* @__PURE__ */ jsxRuntime.jsx("ul", { children: localesForDocument.map((name) => /* @__PURE__ */ jsxRuntime.jsx(designSystem.Box, { padding: 3, tag: "li", children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Typography, { children: name }) }, name)) }) })
705
1252
  ] });
706
1253
  };
707
- const Button = styled__default.default.button`
708
- width: 100%;
709
-
710
- svg {
711
- > g,
712
- path {
713
- fill: ${({ theme }) => theme.colors.neutral500};
714
- }
715
- }
716
- &:hover {
717
- svg {
718
- > g,
719
- path {
720
- fill: ${({ theme }) => theme.colors.neutral600};
721
- }
722
- }
723
- }
724
- &:active {
725
- svg {
726
- > g,
727
- path {
728
- fill: ${({ theme }) => theme.colors.neutral400};
729
- }
730
- }
731
- }
732
- `;
733
- const ActionWrapper = styled__default.default(designSystem.Flex)`
734
- svg {
735
- height: 0.4rem;
736
- }
737
- `;
738
1254
  const addColumnToTableHook = ({ displayedHeaders, layout }) => {
739
1255
  const { options } = layout;
740
1256
  const isFieldLocalized = doesPluginOptionsHaveI18nLocalized(options) ? options.i18n.localized : false;
@@ -763,18 +1279,11 @@ const addColumnToTableHook = ({ displayedHeaders, layout }) => {
763
1279
  const addLocaleToReleasesHook = ({ displayedHeaders = [] }) => {
764
1280
  return {
765
1281
  displayedHeaders: [
766
- // TODO: Fix when migrating to v5
767
- // ...displayedHeaders,
1282
+ ...displayedHeaders,
768
1283
  {
769
- key: "__locale__",
770
- fieldSchema: { type: "string" },
771
- metadatas: {
772
- label: {
773
- id: "content-releases.page.ReleaseDetails.table.header.label.locale",
774
- defaultMessage: "locale"
775
- },
776
- searchable: false,
777
- sortable: false
1284
+ label: {
1285
+ id: "content-releases.page.ReleaseDetails.table.header.label.locale",
1286
+ defaultMessage: "locale"
778
1287
  },
779
1288
  name: "locale"
780
1289
  }
@@ -922,8 +1431,6 @@ const index = {
922
1431
  app.addRBACMiddleware([localeMiddleware]);
923
1432
  app.registerPlugin({
924
1433
  id: pluginId,
925
- initializer: Initializer,
926
- isReady: false,
927
1434
  name: pluginId
928
1435
  });
929
1436
  },
@@ -941,16 +1448,21 @@ const index = {
941
1448
  },
942
1449
  id: "internationalization",
943
1450
  to: "internationalization",
944
- Component: () => Promise.resolve().then(() => require("./SettingsPage-B73NO_8-.js")).then((mod) => ({ default: mod.ProtectedSettingsPage })),
1451
+ Component: () => Promise.resolve().then(() => require("./SettingsPage-kMDCxWLw.js")).then((mod) => ({ default: mod.ProtectedSettingsPage })),
945
1452
  permissions: PERMISSIONS.accessMain
946
1453
  });
947
1454
  const contentManager = app.getPlugin("content-manager");
948
- contentManager.apis.addDocumentHeaderAction([LocalePickerAction]);
1455
+ contentManager.apis.addDocumentHeaderAction([LocalePickerAction, FillFromAnotherLocaleAction]);
949
1456
  contentManager.apis.addDocumentAction((actions) => {
950
1457
  const indexOfDeleteAction = actions.findIndex((action) => action.type === "delete");
951
1458
  actions.splice(indexOfDeleteAction, 0, DeleteLocaleAction);
952
1459
  return actions;
953
1460
  });
1461
+ contentManager.apis.addDocumentAction((actions) => {
1462
+ actions.splice(2, 0, BulkLocalePublishAction);
1463
+ actions.splice(5, 0, BulkLocaleUnpublishAction);
1464
+ return actions;
1465
+ });
954
1466
  contentManager.injectComponent("listView", "actions", {
955
1467
  name: "i18n-locale-filter",
956
1468
  Component: LocalePicker
@@ -1054,7 +1566,7 @@ const index = {
1054
1566
  async registerTrads({ locales }) {
1055
1567
  const importedTrads = await Promise.all(
1056
1568
  locales.map((locale) => {
1057
- return __variableDynamicImportRuntimeHelper(/* @__PURE__ */ Object.assign({ "./translations/de.json": () => Promise.resolve().then(() => require("./de-DtWiGdHl.js")), "./translations/dk.json": () => Promise.resolve().then(() => require("./dk-D8C-casx.js")), "./translations/en.json": () => Promise.resolve().then(() => require("./en-BuBc6LKZ.js")), "./translations/es.json": () => Promise.resolve().then(() => require("./es-DS-XFGSw.js")), "./translations/fr.json": () => Promise.resolve().then(() => require("./fr-BTjekDpq.js")), "./translations/ko.json": () => Promise.resolve().then(() => require("./ko-DmcGUBQ3.js")), "./translations/pl.json": () => Promise.resolve().then(() => require("./pl-Cn5RYonZ.js")), "./translations/ru.json": () => Promise.resolve().then(() => require("./ru-BMBgVL3s.js")), "./translations/tr.json": () => Promise.resolve().then(() => require("./tr-CarUU76c.js")), "./translations/zh-Hans.json": () => Promise.resolve().then(() => require("./zh-Hans-DSHIXAa3.js")), "./translations/zh.json": () => Promise.resolve().then(() => require("./zh-CukOviB0.js")) }), `./translations/${locale}.json`).then(({ default: data }) => {
1569
+ return __variableDynamicImportRuntimeHelper(/* @__PURE__ */ Object.assign({ "./translations/de.json": () => Promise.resolve().then(() => require("./de-DtWiGdHl.js")), "./translations/dk.json": () => Promise.resolve().then(() => require("./dk-D8C-casx.js")), "./translations/en.json": () => Promise.resolve().then(() => require("./en-B6327hMz.js")), "./translations/es.json": () => Promise.resolve().then(() => require("./es-DS-XFGSw.js")), "./translations/fr.json": () => Promise.resolve().then(() => require("./fr-BTjekDpq.js")), "./translations/ko.json": () => Promise.resolve().then(() => require("./ko-DmcGUBQ3.js")), "./translations/pl.json": () => Promise.resolve().then(() => require("./pl-Cn5RYonZ.js")), "./translations/ru.json": () => Promise.resolve().then(() => require("./ru-BMBgVL3s.js")), "./translations/tr.json": () => Promise.resolve().then(() => require("./tr-CarUU76c.js")), "./translations/zh-Hans.json": () => Promise.resolve().then(() => require("./zh-Hans-DSHIXAa3.js")), "./translations/zh.json": () => Promise.resolve().then(() => require("./zh-CukOviB0.js")) }), `./translations/${locale}.json`).then(({ default: data }) => {
1058
1570
  return {
1059
1571
  data: prefixPluginTranslations(data, pluginId),
1060
1572
  locale
@@ -1073,10 +1585,9 @@ const index = {
1073
1585
  exports.PERMISSIONS = PERMISSIONS;
1074
1586
  exports.getTranslation = getTranslation;
1075
1587
  exports.index = index;
1076
- exports.isBaseQueryError = isBaseQueryError;
1077
1588
  exports.useCreateLocaleMutation = useCreateLocaleMutation;
1078
1589
  exports.useDeleteLocaleMutation = useDeleteLocaleMutation;
1079
1590
  exports.useGetDefaultLocalesQuery = useGetDefaultLocalesQuery;
1080
1591
  exports.useGetLocalesQuery = useGetLocalesQuery;
1081
1592
  exports.useUpdateLocaleMutation = useUpdateLocaleMutation;
1082
- //# sourceMappingURL=index-kcYwoCdE.js.map
1593
+ //# sourceMappingURL=index-Dncj9Inq.js.map