@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
@@ -1,17 +1,17 @@
1
1
  import get from "lodash/get";
2
2
  import * as yup from "yup";
3
- import { jsxs, Fragment, jsx } from "react/jsx-runtime";
3
+ import { jsxs, jsx, Fragment } from "react/jsx-runtime";
4
4
  import * as React from "react";
5
- import { Typography, Checkbox, Dialog, DialogBody, Flex, DialogFooter, Button as Button$1, Status, SingleSelect, SingleSelectOption, VisuallyHidden, useCollator, Popover, Box } from "@strapi/design-system";
6
- import { WarningCircle, Trash, Earth, EarthStriked, CaretDown } from "@strapi/icons";
5
+ import { Typography, Dialog, Field, Checkbox, Flex, Button, Modal, Box, Status, IconButton, Tooltip, SingleSelect, SingleSelectOption, VisuallyHidden, useCollator, Popover } from "@strapi/design-system";
6
+ import { WarningCircle, Pencil, CrossCircle, CheckCircle, ArrowsCounterClockwise, Trash, Download, ListPlus, Cross, Earth, EarthStriked, CaretDown } from "@strapi/icons";
7
7
  import { useIntl } from "react-intl";
8
- import styled from "styled-components";
9
- import { useAuth, getFetchClient, useQueryParams, useNotification } from "@strapi/admin/strapi-admin";
10
- import { unstable_useDocument, unstable_useDocumentActions } from "@strapi/content-manager/strapi-admin";
11
- import { useParams, useNavigate, matchPath } from "react-router-dom";
12
- import { createApi } from "@reduxjs/toolkit/query/react";
13
- import { isAxiosError } from "axios";
8
+ import { styled } from "styled-components";
9
+ import { skipToken } from "@reduxjs/toolkit/query";
10
+ import { useAuth, adminApi, useTable, Table, useQueryParams, useForm, useNotification, useAPIErrorHandler } from "@strapi/admin/strapi-admin";
11
+ import { unstable_useDocument, unstable_useDocumentActions, buildValidParams } from "@strapi/content-manager/strapi-admin";
12
+ import { useParams, Link, useNavigate, matchPath } from "react-router-dom";
14
13
  import * as qs from "qs";
14
+ import { stringify } from "qs";
15
15
  import omit from "lodash/omit";
16
16
  const __variableDynamicImportRuntimeHelper = (glob, path) => {
17
17
  const v = glob[path];
@@ -48,9 +48,7 @@ const CheckboxConfirmation = ({
48
48
  };
49
49
  const handleConfirm = () => {
50
50
  onChange({ target: { name, value: false, type: "checkbox" } });
51
- setIsOpen(false);
52
51
  };
53
- const handleToggle = () => setIsOpen((prev) => !prev);
54
52
  const label = intlLabel.id ? formatMessage(
55
53
  { id: intlLabel.id, defaultMessage: intlLabel.defaultMessage },
56
54
  { ...intlLabel.values }
@@ -59,43 +57,36 @@ const CheckboxConfirmation = ({
59
57
  { id: description.id, defaultMessage: description.defaultMessage },
60
58
  { ...description.values }
61
59
  ) : "";
62
- return /* @__PURE__ */ jsxs(Fragment, { children: [
63
- /* @__PURE__ */ jsx(
64
- Checkbox,
65
- {
66
- hint,
67
- id: name,
68
- name,
69
- onValueChange: handleChange,
70
- value,
71
- type: "checkbox",
72
- children: label
73
- }
74
- ),
75
- isOpen && /* @__PURE__ */ jsxs(Dialog, { onClose: handleToggle, title: "Confirmation", isOpen, children: [
76
- /* @__PURE__ */ jsx(DialogBody, { icon: /* @__PURE__ */ jsx(WarningCircle, {}), children: /* @__PURE__ */ jsxs(Flex, { direction: "column", alignItems: "stretch", gap: 2, children: [
77
- /* @__PURE__ */ jsx(Flex, { justifyContent: "center", children: /* @__PURE__ */ jsx(TextAlignTypography, { id: "confirm-description", children: formatMessage({
60
+ return /* @__PURE__ */ jsxs(Dialog.Root, { open: isOpen, onOpenChange: setIsOpen, children: [
61
+ /* @__PURE__ */ jsxs(Field.Root, { hint, name, children: [
62
+ /* @__PURE__ */ jsx(Checkbox, { onCheckedChange: handleChange, checked: value, children: label }),
63
+ /* @__PURE__ */ jsx(Field.Hint, {})
64
+ ] }),
65
+ /* @__PURE__ */ jsxs(Dialog.Content, { children: [
66
+ /* @__PURE__ */ jsx(Dialog.Header, { children: formatMessage({
67
+ id: getTranslation("CheckboxConfirmation.Modal.title"),
68
+ defaultMessage: "Disable localization"
69
+ }) }),
70
+ /* @__PURE__ */ jsx(Dialog.Body, { icon: /* @__PURE__ */ jsx(WarningCircle, {}), children: /* @__PURE__ */ jsxs(Flex, { direction: "column", alignItems: "stretch", gap: 2, children: [
71
+ /* @__PURE__ */ jsx(Flex, { justifyContent: "center", children: /* @__PURE__ */ jsx(TextAlignTypography, { children: formatMessage({
78
72
  id: getTranslation("CheckboxConfirmation.Modal.content"),
79
73
  defaultMessage: "Disabling localization will engender the deletion of all your content but the one associated to your default locale (if existing)."
80
74
  }) }) }),
81
- /* @__PURE__ */ jsx(Flex, { justifyContent: "center", children: /* @__PURE__ */ jsx(Typography, { fontWeight: "semiBold", id: "confirm-description", children: formatMessage({
75
+ /* @__PURE__ */ jsx(Flex, { justifyContent: "center", children: /* @__PURE__ */ jsx(Typography, { fontWeight: "semiBold", children: formatMessage({
82
76
  id: getTranslation("CheckboxConfirmation.Modal.body"),
83
77
  defaultMessage: "Do you want to disable it?"
84
78
  }) }) })
85
79
  ] }) }),
86
- /* @__PURE__ */ jsx(
87
- DialogFooter,
88
- {
89
- startAction: /* @__PURE__ */ jsx(Button$1, { onClick: handleToggle, variant: "tertiary", children: formatMessage({
90
- id: "components.popUpWarning.button.cancel",
91
- defaultMessage: "No, cancel"
92
- }) }),
93
- endAction: /* @__PURE__ */ jsx(Button$1, { variant: "danger-light", onClick: handleConfirm, children: formatMessage({
94
- id: getTranslation("CheckboxConfirmation.Modal.button-confirm"),
95
- defaultMessage: "Yes, disable"
96
- }) })
97
- }
98
- )
80
+ /* @__PURE__ */ jsxs(Dialog.Footer, { children: [
81
+ /* @__PURE__ */ jsx(Dialog.Cancel, { children: /* @__PURE__ */ jsx(Button, { variant: "tertiary", children: formatMessage({
82
+ id: "components.popUpWarning.button.cancel",
83
+ defaultMessage: "No, cancel"
84
+ }) }) }),
85
+ /* @__PURE__ */ jsx(Dialog.Action, { children: /* @__PURE__ */ jsx(Button, { variant: "danger-light", onClick: handleConfirm, children: formatMessage({
86
+ id: getTranslation("CheckboxConfirmation.Modal.button-confirm"),
87
+ defaultMessage: "Yes, disable"
88
+ }) }) })
89
+ ] })
99
90
  ] })
100
91
  ] });
101
92
  };
@@ -146,7 +137,7 @@ const useI18n = () => {
146
137
  model: params.slug
147
138
  },
148
139
  {
149
- skip: !params.slug || !params.collectionType
140
+ skip: true
150
141
  }
151
142
  );
152
143
  if (doesPluginOptionsHaveI18nLocalized(schema?.pluginOptions)) {
@@ -160,64 +151,8 @@ const useI18n = () => {
160
151
  ...actions
161
152
  };
162
153
  };
163
- const axiosBaseQuery = () => async (query, { signal }) => {
164
- try {
165
- const { get: get2, post, del, put } = getFetchClient();
166
- if (typeof query === "string") {
167
- const result = await get2(query, { signal });
168
- return { data: result.data };
169
- } else {
170
- const { url, method = "GET", data, config } = query;
171
- if (method === "POST") {
172
- const result2 = await post(url, data, { ...config, signal });
173
- return { data: result2.data };
174
- }
175
- if (method === "DELETE") {
176
- const result2 = await del(url, { ...config, signal });
177
- return { data: result2.data };
178
- }
179
- if (method === "PUT") {
180
- const result2 = await put(url, data, { ...config, signal });
181
- return { data: result2.data };
182
- }
183
- const result = await get2(url, { ...config, signal });
184
- return { data: result.data };
185
- }
186
- } catch (err) {
187
- if (isAxiosError(err)) {
188
- if (typeof err.response?.data === "object" && err.response?.data !== null && "error" in err.response?.data) {
189
- return { data: void 0, error: err.response?.data.error };
190
- } else {
191
- return {
192
- data: void 0,
193
- error: {
194
- name: "UnknownError",
195
- message: "There was an unknown error response from the API",
196
- details: err.response?.data,
197
- status: err.response?.status
198
- }
199
- };
200
- }
201
- }
202
- const error = err;
203
- return {
204
- data: void 0,
205
- error: {
206
- name: error.name,
207
- message: error.message,
208
- stack: error.stack
209
- }
210
- };
211
- }
212
- };
213
- const isBaseQueryError = (error) => {
214
- return error.name !== void 0;
215
- };
216
- const i18nApi = createApi({
217
- reducerPath: "i18nApi",
218
- baseQuery: axiosBaseQuery(),
219
- tagTypes: ["Locale"],
220
- endpoints: () => ({})
154
+ const i18nApi = adminApi.enhanceEndpoints({
155
+ addTagTypes: ["Locale"]
221
156
  });
222
157
  const localesApi = i18nApi.injectEndpoints({
223
158
  endpoints: (builder) => ({
@@ -266,6 +201,295 @@ const {
266
201
  useGetDefaultLocalesQuery,
267
202
  useUpdateLocaleMutation
268
203
  } = localesApi;
204
+ const relationsApi = i18nApi.injectEndpoints({
205
+ overrideExisting: true,
206
+ endpoints: (builder) => ({
207
+ getManyDraftRelationCount: builder.query({
208
+ query: ({ model, ...params }) => ({
209
+ url: `/content-manager/collection-types/${model}/actions/countManyEntriesDraftRelations`,
210
+ method: "GET",
211
+ config: {
212
+ params
213
+ }
214
+ }),
215
+ transformResponse: (response) => response.data
216
+ })
217
+ })
218
+ });
219
+ const { useGetManyDraftRelationCountQuery } = relationsApi;
220
+ const cleanData = (data, schema, components) => {
221
+ const cleanedData = removeFields(data, [
222
+ "createdAt",
223
+ "createdBy",
224
+ "updatedAt",
225
+ "updatedBy",
226
+ "id",
227
+ "documentId",
228
+ "publishedAt",
229
+ "strapi_stage",
230
+ "strapi_assignee",
231
+ "locale"
232
+ ]);
233
+ const cleanedDataWithoutPasswordAndRelation = recursiveRemoveFieldTypes(
234
+ cleanedData,
235
+ schema,
236
+ components,
237
+ ["relation", "password"]
238
+ );
239
+ return cleanedDataWithoutPasswordAndRelation;
240
+ };
241
+ const removeFields = (data, fields) => {
242
+ return Object.keys(data).reduce((acc, current) => {
243
+ if (fields.includes(current)) {
244
+ return acc;
245
+ }
246
+ acc[current] = data[current];
247
+ return acc;
248
+ }, {});
249
+ };
250
+ const recursiveRemoveFieldTypes = (data, schema, components, fields) => {
251
+ return Object.keys(data).reduce((acc, current) => {
252
+ const attribute = schema.attributes[current] ?? { type: void 0 };
253
+ if (fields.includes(attribute.type)) {
254
+ return acc;
255
+ }
256
+ if (attribute.type === "dynamiczone") {
257
+ acc[current] = data[current].map((componentValue, index2) => {
258
+ const { id: _, ...rest } = recursiveRemoveFieldTypes(
259
+ componentValue,
260
+ components[componentValue.__component],
261
+ components,
262
+ fields
263
+ );
264
+ return {
265
+ ...rest,
266
+ __temp_key__: index2 + 1
267
+ };
268
+ });
269
+ } else if (attribute.type === "component") {
270
+ const { repeatable, component } = attribute;
271
+ if (repeatable) {
272
+ acc[current] = (data[current] ?? []).map((compoData, index2) => {
273
+ const { id: _, ...rest } = recursiveRemoveFieldTypes(
274
+ compoData,
275
+ components[component],
276
+ components,
277
+ fields
278
+ );
279
+ return {
280
+ ...rest,
281
+ __temp_key__: index2 + 1
282
+ };
283
+ });
284
+ } else {
285
+ const { id: _, ...rest } = recursiveRemoveFieldTypes(
286
+ data[current] ?? {},
287
+ components[component],
288
+ components,
289
+ fields
290
+ );
291
+ acc[current] = rest;
292
+ }
293
+ } else {
294
+ acc[current] = data[current];
295
+ }
296
+ return acc;
297
+ }, {});
298
+ };
299
+ const isErrorMessageDescriptor = (object) => {
300
+ return typeof object === "object" && object !== null && "id" in object && "defaultMessage" in object;
301
+ };
302
+ const EntryValidationText = ({
303
+ status = "draft",
304
+ validationErrors,
305
+ action
306
+ }) => {
307
+ const { formatMessage } = useIntl();
308
+ const getErrorStr = (key, value) => {
309
+ if (typeof value === "string") {
310
+ return `${key}: ${value}`;
311
+ } else if (isErrorMessageDescriptor(value)) {
312
+ return `${key}: ${formatMessage(value)}`;
313
+ } else if (Array.isArray(value)) {
314
+ return value.map((v) => getErrorStr(key, v)).join(" ");
315
+ } else if (typeof value === "object" && !Array.isArray(value)) {
316
+ return Object.entries(value).map(([k, v]) => getErrorStr(k, v)).join(" ");
317
+ } else {
318
+ return "";
319
+ }
320
+ };
321
+ if (validationErrors) {
322
+ const validationErrorsMessages = Object.entries(validationErrors).map(([key, value]) => {
323
+ return getErrorStr(key, value);
324
+ }).join(" ");
325
+ return /* @__PURE__ */ jsxs(Flex, { gap: 2, children: [
326
+ /* @__PURE__ */ jsx(CrossCircle, { fill: "danger600" }),
327
+ /* @__PURE__ */ jsx(Tooltip, { label: validationErrorsMessages, children: /* @__PURE__ */ jsx(
328
+ Typography,
329
+ {
330
+ maxWidth: "30rem",
331
+ textColor: "danger600",
332
+ variant: "omega",
333
+ fontWeight: "semiBold",
334
+ ellipsis: true,
335
+ children: validationErrorsMessages
336
+ }
337
+ ) })
338
+ ] });
339
+ }
340
+ const getStatusMessage = () => {
341
+ if (action === "bulk-publish") {
342
+ if (status === "published") {
343
+ return {
344
+ icon: /* @__PURE__ */ jsx(CheckCircle, { fill: "success600" }),
345
+ text: formatMessage({
346
+ id: "content-manager.bulk-publish.already-published",
347
+ defaultMessage: "Already Published"
348
+ }),
349
+ textColor: "success600",
350
+ fontWeight: "bold"
351
+ };
352
+ } else if (status === "modified") {
353
+ return {
354
+ icon: /* @__PURE__ */ jsx(ArrowsCounterClockwise, { fill: "alternative600" }),
355
+ text: formatMessage({
356
+ id: "app.utils.ready-to-publish-changes",
357
+ defaultMessage: "Ready to publish changes"
358
+ })
359
+ };
360
+ } else {
361
+ return {
362
+ icon: /* @__PURE__ */ jsx(CheckCircle, { fill: "success600" }),
363
+ text: formatMessage({
364
+ id: "app.utils.ready-to-publish",
365
+ defaultMessage: "Ready to publish"
366
+ })
367
+ };
368
+ }
369
+ } else {
370
+ if (status === "draft") {
371
+ return {
372
+ icon: /* @__PURE__ */ jsx(CheckCircle, { fill: "success600" }),
373
+ text: formatMessage({
374
+ id: "content-manager.bulk-unpublish.already-unpublished",
375
+ defaultMessage: "Already Unpublished"
376
+ }),
377
+ textColor: "success600",
378
+ fontWeight: "bold"
379
+ };
380
+ } else {
381
+ return {
382
+ icon: /* @__PURE__ */ jsx(CheckCircle, { fill: "success600" }),
383
+ text: formatMessage({
384
+ id: "app.utils.ready-to-unpublish-changes",
385
+ defaultMessage: "Ready to unpublish"
386
+ }),
387
+ textColor: "success600",
388
+ fontWeight: "bold"
389
+ };
390
+ }
391
+ }
392
+ };
393
+ const { icon, text, textColor = "success600", fontWeight = "normal" } = getStatusMessage();
394
+ return /* @__PURE__ */ jsxs(Flex, { gap: 2, children: [
395
+ icon,
396
+ /* @__PURE__ */ jsx(Typography, { textColor, fontWeight, children: text })
397
+ ] });
398
+ };
399
+ const BoldChunk = (chunks) => /* @__PURE__ */ jsx(Typography, { fontWeight: "bold", children: chunks });
400
+ const BulkLocaleActionModal = ({
401
+ headers,
402
+ rows,
403
+ localesMetadata,
404
+ validationErrors = {},
405
+ action
406
+ }) => {
407
+ const { formatMessage } = useIntl();
408
+ const selectedRows = useTable(
409
+ "BulkLocaleActionModal",
410
+ (state) => state.selectedRows
411
+ );
412
+ const getFormattedCountMessage = () => {
413
+ const currentStatusByLocale = rows.reduce((acc, { locale, status }) => {
414
+ acc[locale] = status;
415
+ return acc;
416
+ }, {});
417
+ const localesWithErrors = Object.keys(validationErrors);
418
+ const publishedCount = selectedRows.filter(
419
+ ({ locale }) => currentStatusByLocale[locale] === "published"
420
+ ).length;
421
+ const draftCount = selectedRows.filter(
422
+ ({ locale }) => (currentStatusByLocale[locale] === "draft" || currentStatusByLocale[locale] === "modified") && !localesWithErrors.includes(locale)
423
+ ).length;
424
+ const withErrorsCount = localesWithErrors.length;
425
+ const messageId = action === "bulk-publish" ? "content-manager.containers.list.selectedEntriesModal.selectedCount.publish" : "content-manager.containers.list.selectedEntriesModal.selectedCount.unpublish";
426
+ 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.";
427
+ return formatMessage(
428
+ {
429
+ id: messageId,
430
+ defaultMessage
431
+ },
432
+ {
433
+ withErrorsCount,
434
+ draftCount,
435
+ publishedCount,
436
+ b: BoldChunk
437
+ }
438
+ );
439
+ };
440
+ return /* @__PURE__ */ jsxs(Modal.Body, { children: [
441
+ /* @__PURE__ */ jsx(Typography, { children: getFormattedCountMessage() }),
442
+ /* @__PURE__ */ jsx(Box, { marginTop: 5, children: /* @__PURE__ */ jsxs(Table.Content, { children: [
443
+ /* @__PURE__ */ jsxs(Table.Head, { children: [
444
+ /* @__PURE__ */ jsx(Table.HeaderCheckboxCell, {}),
445
+ headers.map((head) => /* @__PURE__ */ jsx(Table.HeaderCell, { ...head }, head.name))
446
+ ] }),
447
+ /* @__PURE__ */ jsx(Table.Body, { children: rows.map(({ locale, status }, index2) => {
448
+ const error = validationErrors?.[locale] ?? null;
449
+ const statusVariant = status === "draft" ? "primary" : status === "published" ? "success" : "alternative";
450
+ return /* @__PURE__ */ jsxs(Table.Row, { children: [
451
+ /* @__PURE__ */ jsx(Table.CheckboxCell, { id: locale, "aria-label": `Select ${locale}` }),
452
+ /* @__PURE__ */ jsx(Table.Cell, { children: /* @__PURE__ */ jsx(Typography, { variant: "sigma", textColor: "neutral600", children: Array.isArray(localesMetadata) ? localesMetadata.find((localeEntry) => localeEntry.code === locale)?.name : locale }) }),
453
+ /* @__PURE__ */ jsx(Table.Cell, { children: /* @__PURE__ */ jsx(Box, { display: "flex", children: /* @__PURE__ */ jsx(
454
+ Status,
455
+ {
456
+ display: "flex",
457
+ paddingLeft: "6px",
458
+ paddingRight: "6px",
459
+ paddingTop: "2px",
460
+ paddingBottom: "2px",
461
+ showBullet: false,
462
+ size: "S",
463
+ variant: statusVariant,
464
+ children: /* @__PURE__ */ jsx(Typography, { tag: "span", variant: "pi", fontWeight: "bold", children: capitalize(status) })
465
+ }
466
+ ) }) }),
467
+ /* @__PURE__ */ jsx(Table.Cell, { children: /* @__PURE__ */ jsx(EntryValidationText, { validationErrors: error, status, action }) }),
468
+ /* @__PURE__ */ jsx(Table.Cell, { children: /* @__PURE__ */ jsx(
469
+ IconButton,
470
+ {
471
+ tag: Link,
472
+ to: {
473
+ search: stringify({ plugins: { i18n: { locale } } })
474
+ },
475
+ label: formatMessage(
476
+ {
477
+ id: getTranslation("Settings.list.actions.edit"),
478
+ defaultMessage: "Edit {name} locale"
479
+ },
480
+ {
481
+ name: locale
482
+ }
483
+ ),
484
+ variant: "ghost",
485
+ children: /* @__PURE__ */ jsx(Pencil, {})
486
+ }
487
+ ) })
488
+ ] }, index2);
489
+ }) })
490
+ ] }) })
491
+ ] });
492
+ };
269
493
  const LocalePickerAction = ({
270
494
  document,
271
495
  meta,
@@ -277,7 +501,13 @@ const LocalePickerAction = ({
277
501
  const [{ query }, setQuery] = useQueryParams();
278
502
  const { hasI18n, canCreate, canRead } = useI18n();
279
503
  const { data: locales = [] } = useGetLocalesQuery();
280
- const { schema } = unstable_useDocument({ model, collectionType, documentId });
504
+ const currentDesiredLocale = query.plugins?.i18n?.locale;
505
+ const { schema } = unstable_useDocument({
506
+ model,
507
+ collectionType,
508
+ documentId,
509
+ params: { locale: currentDesiredLocale }
510
+ });
281
511
  const handleSelect = React.useCallback(
282
512
  (value) => {
283
513
  setQuery({
@@ -295,21 +525,20 @@ const LocalePickerAction = ({
295
525
  if (!Array.isArray(locales) || !hasI18n) {
296
526
  return;
297
527
  }
298
- const currentDesiredLocale = query.plugins?.i18n?.locale;
299
528
  const doesLocaleExist = locales.find((loc) => loc.code === currentDesiredLocale);
300
529
  const defaultLocale = locales.find((locale) => locale.isDefault);
301
530
  if (!doesLocaleExist && defaultLocale?.code) {
302
531
  handleSelect(defaultLocale.code);
303
532
  }
304
- }, [handleSelect, hasI18n, locales, query.plugins?.i18n?.locale]);
305
- if (!hasI18n || !Array.isArray(locales) || locales.length === 0) {
306
- return null;
307
- }
308
- const currentLocale = query.plugins?.i18n?.locale || locales.find((loc) => loc.isDefault)?.code;
533
+ }, [handleSelect, hasI18n, locales, currentDesiredLocale]);
534
+ const currentLocale = Array.isArray(locales) ? locales.find((locale) => locale.code === currentDesiredLocale)?.code : void 0;
309
535
  const allCurrentLocales = [
310
536
  { status: getDocumentStatus(document, meta), locale: currentLocale },
311
537
  ...meta?.availableLocales ?? []
312
538
  ];
539
+ if (!hasI18n || !Array.isArray(locales) || locales.length === 0) {
540
+ return null;
541
+ }
313
542
  return {
314
543
  label: formatMessage({
315
544
  id: getTranslation("Settings.locales.modal.locales.label"),
@@ -321,7 +550,7 @@ const LocalePickerAction = ({
321
550
  );
322
551
  const status = currentLocaleDoc?.status ?? "draft";
323
552
  const permissionsToCheck = currentLocaleDoc ? canCreate : canRead;
324
- const statusVariant = status === "draft" ? "primary" : status === "published" ? "success" : "alternative";
553
+ const statusVariant = status === "draft" ? "secondary" : status === "published" ? "success" : "alternative";
325
554
  return {
326
555
  disabled: !permissionsToCheck.includes(locale.code),
327
556
  value: locale.code,
@@ -337,7 +566,7 @@ const LocalePickerAction = ({
337
566
  showBullet: false,
338
567
  size: "S",
339
568
  variant: statusVariant,
340
- children: /* @__PURE__ */ jsx(Typography, { as: "span", variant: "pi", fontWeight: "bold", children: capitalize(status) })
569
+ children: /* @__PURE__ */ jsx(Typography, { tag: "span", variant: "pi", fontWeight: "bold", children: capitalize(status) })
341
570
  }
342
571
  ) : null
343
572
  };
@@ -357,6 +586,95 @@ const getDocumentStatus = (document, meta) => {
357
586
  }
358
587
  return docStatus;
359
588
  };
589
+ const FillFromAnotherLocaleAction = ({
590
+ documentId,
591
+ meta,
592
+ model,
593
+ collectionType
594
+ }) => {
595
+ const { formatMessage } = useIntl();
596
+ const [{ query }] = useQueryParams();
597
+ const currentDesiredLocale = query.plugins?.i18n?.locale;
598
+ const [localeSelected, setLocaleSelected] = React.useState(null);
599
+ const setValues = useForm("FillFromAnotherLocale", (state) => state.setValues);
600
+ const { getDocument } = unstable_useDocumentActions();
601
+ const { schema, components } = unstable_useDocument({
602
+ model,
603
+ documentId,
604
+ collectionType,
605
+ params: { locale: currentDesiredLocale }
606
+ });
607
+ const { data: locales = [] } = useGetLocalesQuery();
608
+ const availableLocales = Array.isArray(locales) ? locales.filter((locale) => meta?.availableLocales.some((l) => l.locale === locale.code)) : [];
609
+ const fillFromLocale = (onClose) => async () => {
610
+ const response = await getDocument({
611
+ collectionType,
612
+ model,
613
+ documentId,
614
+ params: { locale: localeSelected }
615
+ });
616
+ if (!response || !schema) {
617
+ return;
618
+ }
619
+ const { data } = response;
620
+ const cleanedData = cleanData(data, schema, components);
621
+ setValues(cleanedData);
622
+ onClose();
623
+ };
624
+ return {
625
+ type: "icon",
626
+ icon: /* @__PURE__ */ jsx(Download, {}),
627
+ disabled: availableLocales.length === 0,
628
+ label: formatMessage({
629
+ id: getTranslation("CMEditViewCopyLocale.copy-text"),
630
+ defaultMessage: "Fill in from another locale"
631
+ }),
632
+ dialog: {
633
+ type: "dialog",
634
+ title: formatMessage({
635
+ id: getTranslation("CMEditViewCopyLocale.dialog.title"),
636
+ defaultMessage: "Confirmation"
637
+ }),
638
+ content: ({ onClose }) => /* @__PURE__ */ jsxs(Fragment, { children: [
639
+ /* @__PURE__ */ jsx(Dialog.Body, { children: /* @__PURE__ */ jsxs(Flex, { direction: "column", gap: 3, children: [
640
+ /* @__PURE__ */ jsx(WarningCircle, { width: "24px", height: "24px", fill: "danger600" }),
641
+ /* @__PURE__ */ jsx(Typography, { textAlign: "center", children: formatMessage({
642
+ id: getTranslation("CMEditViewCopyLocale.dialog.body"),
643
+ defaultMessage: "Your current content will be erased and filled by the content of the selected locale:"
644
+ }) }),
645
+ /* @__PURE__ */ jsxs(Field.Root, { width: "100%", children: [
646
+ /* @__PURE__ */ jsx(Field.Label, { children: formatMessage({
647
+ id: getTranslation("CMEditViewCopyLocale.dialog.field.label"),
648
+ defaultMessage: "Locale"
649
+ }) }),
650
+ /* @__PURE__ */ jsx(
651
+ SingleSelect,
652
+ {
653
+ value: localeSelected,
654
+ placeholder: formatMessage({
655
+ id: getTranslation("CMEditViewCopyLocale.dialog.field.placeholder"),
656
+ defaultMessage: "Select one locale..."
657
+ }),
658
+ onChange: (value) => setLocaleSelected(value),
659
+ children: availableLocales.map((locale) => /* @__PURE__ */ jsx(SingleSelectOption, { value: locale.code, children: locale.name }, locale.code))
660
+ }
661
+ )
662
+ ] })
663
+ ] }) }),
664
+ /* @__PURE__ */ jsx(Dialog.Footer, { children: /* @__PURE__ */ jsxs(Flex, { gap: 2, width: "100%", children: [
665
+ /* @__PURE__ */ jsx(Button, { flex: "auto", variant: "tertiary", onClick: onClose, children: formatMessage({
666
+ id: getTranslation("CMEditViewCopyLocale.cancel-text"),
667
+ defaultMessage: "No, cancel"
668
+ }) }),
669
+ /* @__PURE__ */ jsx(Button, { flex: "auto", variant: "success", onClick: fillFromLocale(onClose), children: formatMessage({
670
+ id: getTranslation("CMEditViewCopyLocale.submit-text"),
671
+ defaultMessage: "Yes, fill in"
672
+ }) })
673
+ ] }) })
674
+ ] })
675
+ }
676
+ };
677
+ };
360
678
  const DeleteLocaleAction = ({
361
679
  document,
362
680
  documentId,
@@ -368,16 +686,23 @@ const DeleteLocaleAction = ({
368
686
  const { toggleNotification } = useNotification();
369
687
  const { delete: deleteAction } = unstable_useDocumentActions();
370
688
  const { hasI18n, canDelete } = useI18n();
689
+ const [{ query }] = useQueryParams();
690
+ const { data: locales = [] } = useGetLocalesQuery();
691
+ const currentDesiredLocale = query.plugins?.i18n?.locale;
692
+ const locale = !("error" in locales) && locales.find((loc) => loc.code === currentDesiredLocale);
371
693
  if (!hasI18n) {
372
694
  return null;
373
695
  }
374
696
  return {
375
697
  disabled: document?.locale && !canDelete.includes(document.locale) || !document || !document.id,
376
698
  position: ["header", "table-row"],
377
- label: formatMessage({
378
- id: getTranslation("actions.delete.label"),
379
- defaultMessage: "Delete locale"
380
- }),
699
+ label: formatMessage(
700
+ {
701
+ id: getTranslation("actions.delete.label"),
702
+ defaultMessage: "Delete entry ({locale})"
703
+ },
704
+ { locale: locale && locale.name }
705
+ ),
381
706
  icon: /* @__PURE__ */ jsx(StyledTrash, {}),
382
707
  variant: "danger",
383
708
  dialog: {
@@ -388,7 +713,7 @@ const DeleteLocaleAction = ({
388
713
  }),
389
714
  content: /* @__PURE__ */ jsxs(Flex, { direction: "column", gap: 2, children: [
390
715
  /* @__PURE__ */ jsx(WarningCircle, { width: "24px", height: "24px", fill: "danger600" }),
391
- /* @__PURE__ */ jsx(Typography, { as: "p", variant: "omega", textAlign: "center", children: formatMessage({
716
+ /* @__PURE__ */ jsx(Typography, { tag: "p", variant: "omega", textAlign: "center", children: formatMessage({
392
717
  id: getTranslation("actions.delete.dialog.body"),
393
718
  defaultMessage: "Are you sure?"
394
719
  }) })
@@ -420,6 +745,262 @@ const DeleteLocaleAction = ({
420
745
  }
421
746
  };
422
747
  };
748
+ const BulkLocaleAction = ({
749
+ document: baseDocument,
750
+ documentId,
751
+ model,
752
+ collectionType,
753
+ action
754
+ }) => {
755
+ const baseLocale = baseDocument?.locale ?? null;
756
+ const [{ query }] = useQueryParams();
757
+ const params = React.useMemo(() => buildValidParams(query), [query]);
758
+ const isOnPublishedTab = query.status === "published";
759
+ const { formatMessage } = useIntl();
760
+ const { hasI18n, canPublish } = useI18n();
761
+ const { toggleNotification } = useNotification();
762
+ const { _unstableFormatAPIError: formatAPIError } = useAPIErrorHandler();
763
+ const [selectedRows, setSelectedRows] = React.useState([]);
764
+ const [isDraftRelationConfirmationOpen, setIsDraftRelationConfirmationOpen] = React.useState(false);
765
+ const { publishMany: publishManyAction, unpublishMany: unpublishManyAction } = unstable_useDocumentActions();
766
+ const {
767
+ document,
768
+ meta: documentMeta,
769
+ schema,
770
+ validate
771
+ } = unstable_useDocument(
772
+ {
773
+ model,
774
+ collectionType,
775
+ documentId,
776
+ params: {
777
+ locale: baseLocale
778
+ }
779
+ },
780
+ {
781
+ skip: !hasI18n || !baseLocale
782
+ }
783
+ );
784
+ const { data: localesMetadata = [] } = useGetLocalesQuery(hasI18n ? void 0 : skipToken);
785
+ const headers = [
786
+ {
787
+ label: formatMessage({
788
+ id: "global.name",
789
+ defaultMessage: "Name"
790
+ }),
791
+ name: "name"
792
+ },
793
+ {
794
+ label: formatMessage({
795
+ id: getTranslation("CMEditViewBulkLocale.status"),
796
+ defaultMessage: "Status"
797
+ }),
798
+ name: "status"
799
+ },
800
+ {
801
+ label: formatMessage({
802
+ id: getTranslation("CMEditViewBulkLocale.publication-status"),
803
+ defaultMessage: "Publication Status"
804
+ }),
805
+ name: "publication-status"
806
+ }
807
+ ];
808
+ const [rows, validationErrors] = React.useMemo(() => {
809
+ if (!document || !documentMeta?.availableLocales) {
810
+ return [[], {}];
811
+ }
812
+ const rowsFromMeta = documentMeta?.availableLocales.map((doc) => {
813
+ const { locale, status } = doc;
814
+ return { locale, status };
815
+ });
816
+ rowsFromMeta.unshift({
817
+ locale: document.locale,
818
+ status: document.status
819
+ });
820
+ const allDocuments = [document, ...documentMeta?.availableLocales ?? []];
821
+ const errors = allDocuments.reduce((errs, document2) => {
822
+ if (!document2) {
823
+ return errs;
824
+ }
825
+ const validation = validate(document2);
826
+ if (validation !== null) {
827
+ errs[document2.locale] = validation;
828
+ }
829
+ return errs;
830
+ }, {});
831
+ return [rowsFromMeta, errors];
832
+ }, [document, documentMeta?.availableLocales, validate]);
833
+ const isBulkPublish = action === "bulk-publish";
834
+ const localesForAction = selectedRows.reduce((acc, selectedRow) => {
835
+ const isValidLocale = (
836
+ // Validation errors are irrelevant if we are trying to unpublish
837
+ !isBulkPublish || !Object.keys(validationErrors).includes(selectedRow.locale)
838
+ );
839
+ const shouldAddLocale = isBulkPublish ? selectedRow.status !== "published" && isValidLocale : selectedRow.status !== "draft" && isValidLocale;
840
+ if (shouldAddLocale) {
841
+ acc.push(selectedRow.locale);
842
+ }
843
+ return acc;
844
+ }, []);
845
+ const enableDraftRelationsCount = false;
846
+ const {
847
+ data: draftRelationsCount = 0,
848
+ isLoading: isDraftRelationsLoading,
849
+ error: isDraftRelationsError
850
+ } = useGetManyDraftRelationCountQuery(
851
+ {
852
+ model,
853
+ documentIds: [documentId],
854
+ locale: localesForAction
855
+ },
856
+ {
857
+ skip: !enableDraftRelationsCount
858
+ }
859
+ );
860
+ React.useEffect(() => {
861
+ if (isDraftRelationsError) {
862
+ toggleNotification({
863
+ type: "danger",
864
+ message: formatAPIError(isDraftRelationsError)
865
+ });
866
+ }
867
+ }, [isDraftRelationsError, toggleNotification, formatAPIError]);
868
+ if (!schema?.options?.draftAndPublish) {
869
+ return null;
870
+ }
871
+ if (!hasI18n) {
872
+ return null;
873
+ }
874
+ if (!documentId) {
875
+ return null;
876
+ }
877
+ const publish = async () => {
878
+ await publishManyAction({
879
+ model,
880
+ documentIds: [documentId],
881
+ params: {
882
+ ...params,
883
+ locale: localesForAction
884
+ }
885
+ });
886
+ setSelectedRows([]);
887
+ };
888
+ const unpublish = async () => {
889
+ await unpublishManyAction({
890
+ model,
891
+ documentIds: [documentId],
892
+ params: {
893
+ ...params,
894
+ locale: localesForAction
895
+ }
896
+ });
897
+ setSelectedRows([]);
898
+ };
899
+ const handleAction = async () => {
900
+ if (draftRelationsCount > 0) {
901
+ setIsDraftRelationConfirmationOpen(true);
902
+ } else if (isBulkPublish) {
903
+ await publish();
904
+ } else {
905
+ await unpublish();
906
+ }
907
+ };
908
+ if (isDraftRelationConfirmationOpen) {
909
+ return {
910
+ label: formatMessage({
911
+ id: "app.components.ConfirmDialog.title",
912
+ defaultMessage: "Confirmation"
913
+ }),
914
+ variant: "danger",
915
+ dialog: {
916
+ onCancel: () => {
917
+ setIsDraftRelationConfirmationOpen(false);
918
+ },
919
+ onConfirm: async () => {
920
+ await publish();
921
+ setIsDraftRelationConfirmationOpen(false);
922
+ },
923
+ type: "dialog",
924
+ title: formatMessage({
925
+ id: getTranslation("actions.publish.dialog.title"),
926
+ defaultMessage: "Confirmation"
927
+ }),
928
+ content: /* @__PURE__ */ jsxs(Flex, { direction: "column", alignItems: "center", gap: 2, children: [
929
+ /* @__PURE__ */ jsx(WarningCircle, { width: "2.4rem", height: "2.4rem", fill: "danger600" }),
930
+ /* @__PURE__ */ jsx(Typography, { textAlign: "center", children: formatMessage({
931
+ id: getTranslation("CMEditViewBulkLocale.draft-relation-warning"),
932
+ defaultMessage: "Some locales are related to draft entries. Publishing them could leave broken links in your app."
933
+ }) }),
934
+ /* @__PURE__ */ jsx(Typography, { textAlign: "center", children: formatMessage({
935
+ id: getTranslation("CMEditViewBulkLocale.continue-confirmation"),
936
+ defaultMessage: "Are you sure you want to continue?"
937
+ }) })
938
+ ] })
939
+ }
940
+ };
941
+ }
942
+ const hasPermission = selectedRows.map(({ locale }) => locale).every((locale) => canPublish.includes(locale));
943
+ return {
944
+ label: formatMessage({
945
+ id: getTranslation(`CMEditViewBulkLocale.${isBulkPublish ? "publish" : "unpublish"}-title`),
946
+ defaultMessage: `${isBulkPublish ? "Publish" : "Unpublish"} Multiple Locales`
947
+ }),
948
+ variant: isBulkPublish ? "secondary" : "danger",
949
+ icon: isBulkPublish ? /* @__PURE__ */ jsx(ListPlus, {}) : /* @__PURE__ */ jsx(Cross, {}),
950
+ disabled: isOnPublishedTab || canPublish.length === 0,
951
+ position: ["panel"],
952
+ dialog: {
953
+ type: "modal",
954
+ title: formatMessage({
955
+ id: getTranslation(`CMEditViewBulkLocale.${isBulkPublish ? "publish" : "unpublish"}-title`),
956
+ defaultMessage: `${isBulkPublish ? "Publish" : "Unpublish"} Multiple Locales`
957
+ }),
958
+ content: () => {
959
+ return /* @__PURE__ */ jsx(
960
+ Table.Root,
961
+ {
962
+ headers,
963
+ rows: rows.map((row) => ({
964
+ ...row,
965
+ id: row.locale
966
+ })),
967
+ selectedRows,
968
+ onSelectedRowsChange: (tableSelectedRows) => setSelectedRows(tableSelectedRows),
969
+ children: /* @__PURE__ */ jsx(
970
+ BulkLocaleActionModal,
971
+ {
972
+ validationErrors,
973
+ headers,
974
+ rows,
975
+ localesMetadata,
976
+ action: action ?? "bulk-publish"
977
+ }
978
+ )
979
+ }
980
+ );
981
+ },
982
+ footer: () => /* @__PURE__ */ jsx(Modal.Footer, { justifyContent: "flex-end", children: /* @__PURE__ */ jsx(
983
+ Button,
984
+ {
985
+ loading: isDraftRelationsLoading,
986
+ disabled: !hasPermission || localesForAction.length === 0,
987
+ variant: "default",
988
+ onClick: handleAction,
989
+ children: formatMessage({
990
+ id: isBulkPublish ? "app.utils.publish" : "app.utils.unpublish",
991
+ defaultMessage: isBulkPublish ? "Publish" : "Unpublish"
992
+ })
993
+ }
994
+ ) })
995
+ }
996
+ };
997
+ };
998
+ const BulkLocalePublishAction = (props) => {
999
+ return BulkLocaleAction({ action: "bulk-publish", ...props });
1000
+ };
1001
+ const BulkLocaleUnpublishAction = (props) => {
1002
+ return BulkLocaleAction({ action: "bulk-unpublish", ...props });
1003
+ };
423
1004
  const StyledTrash = styled(Trash)`
424
1005
  path {
425
1006
  fill: currentColor;
@@ -476,13 +1057,6 @@ const UnpublishModalAdditionalInfo = () => {
476
1057
  }
477
1058
  ) });
478
1059
  };
479
- const Initializer = ({ setPlugin }) => {
480
- const setPluginRef = React.useRef(setPlugin);
481
- React.useEffect(() => {
482
- setPluginRef.current(pluginId);
483
- }, []);
484
- return null;
485
- };
486
1060
  const LocalePicker = () => {
487
1061
  const { formatMessage } = useIntl();
488
1062
  const [{ query }, setQuery] = useQueryParams();
@@ -542,7 +1116,7 @@ const PERMISSIONS = {
542
1116
  read: [{ action: "plugin::i18n.locale.read", subject: null }]
543
1117
  };
544
1118
  const mutateEditViewHook = ({ layout }) => {
545
- if ("i18n" in layout.options && typeof layout.options.i18n === "object" && layout.options.i18n !== null && "localized" in layout.options.i18n && !layout.options.i18n.localized) {
1119
+ if (!("i18n" in layout.options) || typeof layout.options.i18n === "object" && layout.options.i18n !== null && "localized" in layout.options.i18n && !layout.options.i18n.localized) {
546
1120
  return { layout };
547
1121
  }
548
1122
  const components = Object.entries(layout.components).reduce(
@@ -587,8 +1161,8 @@ const doesFieldHaveI18nPluginOpt = (pluginOpts) => {
587
1161
  };
588
1162
  const LabelAction = ({ title, icon }) => {
589
1163
  const { formatMessage } = useIntl();
590
- return /* @__PURE__ */ jsxs(Span, { as: "span", children: [
591
- /* @__PURE__ */ jsx(VisuallyHidden, { as: "span", children: `(${formatMessage(title)})` }),
1164
+ return /* @__PURE__ */ jsxs(Span, { tag: "span", children: [
1165
+ /* @__PURE__ */ jsx(VisuallyHidden, { tag: "span", children: formatMessage(title) }),
592
1166
  React.cloneElement(icon, {
593
1167
  "aria-hidden": true,
594
1168
  focusable: false
@@ -623,13 +1197,7 @@ const LocaleListCell = ({
623
1197
  }
624
1198
  });
625
1199
  const { locale: language } = useIntl();
626
- const [visible, setVisible] = React.useState(false);
627
- const buttonRef = React.useRef(null);
628
1200
  const { data: locales = [] } = useGetLocalesQuery();
629
- const handleTogglePopover = (e) => {
630
- e.stopPropagation();
631
- setVisible((prev) => !prev);
632
- };
633
1201
  const formatter = useCollator(language, {
634
1202
  sensitivity: "base"
635
1203
  });
@@ -651,64 +1219,14 @@ const LocaleListCell = ({
651
1219
  }
652
1220
  return locale.name;
653
1221
  }).toSorted((a, b) => formatter.compare(a, b));
654
- return /* @__PURE__ */ jsxs(Button, { type: "button", onClick: handleTogglePopover, ref: buttonRef, children: [
655
- /* @__PURE__ */ jsxs(
656
- ActionWrapper,
657
- {
658
- minWidth: "100%",
659
- alignItems: "center",
660
- justifyContent: "center",
661
- height: "3.2rem",
662
- width: "3.2rem",
663
- children: [
664
- /* @__PURE__ */ jsx(Typography, { textColor: "neutral800", ellipsis: true, children: localesForDocument.join(", ") }),
665
- /* @__PURE__ */ jsx(Flex, { children: /* @__PURE__ */ jsx(CaretDown, {}) })
666
- ]
667
- }
668
- ),
669
- visible && /* @__PURE__ */ jsx(
670
- Popover,
671
- {
672
- onDismiss: () => setVisible(false),
673
- source: buttonRef,
674
- spacing: 16,
675
- centered: true,
676
- children: /* @__PURE__ */ jsx("ul", { children: localesForDocument.map((name) => /* @__PURE__ */ jsx(Box, { padding: 3, as: "li", children: /* @__PURE__ */ jsx(Typography, { children: name }) }, name)) })
677
- }
678
- )
1222
+ return /* @__PURE__ */ jsxs(Popover.Root, { children: [
1223
+ /* @__PURE__ */ jsx(Popover.Trigger, { children: /* @__PURE__ */ jsx(Button, { variant: "ghost", type: "button", onClick: (e) => e.stopPropagation(), children: /* @__PURE__ */ jsxs(Flex, { minWidth: "100%", alignItems: "center", justifyContent: "center", fontWeight: "regular", children: [
1224
+ /* @__PURE__ */ jsx(Typography, { textColor: "neutral800", ellipsis: true, marginRight: 2, children: localesForDocument.join(", ") }),
1225
+ /* @__PURE__ */ jsx(Flex, { children: /* @__PURE__ */ jsx(CaretDown, { width: "1.2rem", height: "1.2rem" }) })
1226
+ ] }) }) }),
1227
+ /* @__PURE__ */ jsx(Popover.Content, { sideOffset: 16, children: /* @__PURE__ */ jsx("ul", { children: localesForDocument.map((name) => /* @__PURE__ */ jsx(Box, { padding: 3, tag: "li", children: /* @__PURE__ */ jsx(Typography, { children: name }) }, name)) }) })
679
1228
  ] });
680
1229
  };
681
- const Button = styled.button`
682
- width: 100%;
683
-
684
- svg {
685
- > g,
686
- path {
687
- fill: ${({ theme }) => theme.colors.neutral500};
688
- }
689
- }
690
- &:hover {
691
- svg {
692
- > g,
693
- path {
694
- fill: ${({ theme }) => theme.colors.neutral600};
695
- }
696
- }
697
- }
698
- &:active {
699
- svg {
700
- > g,
701
- path {
702
- fill: ${({ theme }) => theme.colors.neutral400};
703
- }
704
- }
705
- }
706
- `;
707
- const ActionWrapper = styled(Flex)`
708
- svg {
709
- height: 0.4rem;
710
- }
711
- `;
712
1230
  const addColumnToTableHook = ({ displayedHeaders, layout }) => {
713
1231
  const { options } = layout;
714
1232
  const isFieldLocalized = doesPluginOptionsHaveI18nLocalized(options) ? options.i18n.localized : false;
@@ -737,18 +1255,11 @@ const addColumnToTableHook = ({ displayedHeaders, layout }) => {
737
1255
  const addLocaleToReleasesHook = ({ displayedHeaders = [] }) => {
738
1256
  return {
739
1257
  displayedHeaders: [
740
- // TODO: Fix when migrating to v5
741
- // ...displayedHeaders,
1258
+ ...displayedHeaders,
742
1259
  {
743
- key: "__locale__",
744
- fieldSchema: { type: "string" },
745
- metadatas: {
746
- label: {
747
- id: "content-releases.page.ReleaseDetails.table.header.label.locale",
748
- defaultMessage: "locale"
749
- },
750
- searchable: false,
751
- sortable: false
1260
+ label: {
1261
+ id: "content-releases.page.ReleaseDetails.table.header.label.locale",
1262
+ defaultMessage: "locale"
752
1263
  },
753
1264
  name: "locale"
754
1265
  }
@@ -896,8 +1407,6 @@ const index = {
896
1407
  app.addRBACMiddleware([localeMiddleware]);
897
1408
  app.registerPlugin({
898
1409
  id: pluginId,
899
- initializer: Initializer,
900
- isReady: false,
901
1410
  name: pluginId
902
1411
  });
903
1412
  },
@@ -915,16 +1424,21 @@ const index = {
915
1424
  },
916
1425
  id: "internationalization",
917
1426
  to: "internationalization",
918
- Component: () => import("./SettingsPage-pvdzbKEM.mjs").then((mod) => ({ default: mod.ProtectedSettingsPage })),
1427
+ Component: () => import("./SettingsPage-BIZrSFGY.mjs").then((mod) => ({ default: mod.ProtectedSettingsPage })),
919
1428
  permissions: PERMISSIONS.accessMain
920
1429
  });
921
1430
  const contentManager = app.getPlugin("content-manager");
922
- contentManager.apis.addDocumentHeaderAction([LocalePickerAction]);
1431
+ contentManager.apis.addDocumentHeaderAction([LocalePickerAction, FillFromAnotherLocaleAction]);
923
1432
  contentManager.apis.addDocumentAction((actions) => {
924
1433
  const indexOfDeleteAction = actions.findIndex((action) => action.type === "delete");
925
1434
  actions.splice(indexOfDeleteAction, 0, DeleteLocaleAction);
926
1435
  return actions;
927
1436
  });
1437
+ contentManager.apis.addDocumentAction((actions) => {
1438
+ actions.splice(2, 0, BulkLocalePublishAction);
1439
+ actions.splice(5, 0, BulkLocaleUnpublishAction);
1440
+ return actions;
1441
+ });
928
1442
  contentManager.injectComponent("listView", "actions", {
929
1443
  name: "i18n-locale-filter",
930
1444
  Component: LocalePicker
@@ -1028,7 +1542,7 @@ const index = {
1028
1542
  async registerTrads({ locales }) {
1029
1543
  const importedTrads = await Promise.all(
1030
1544
  locales.map((locale) => {
1031
- return __variableDynamicImportRuntimeHelper(/* @__PURE__ */ Object.assign({ "./translations/de.json": () => import("./de-9eCAqqrB.mjs"), "./translations/dk.json": () => import("./dk-2qBjxt-P.mjs"), "./translations/en.json": () => import("./en-CnrTsjWS.mjs"), "./translations/es.json": () => import("./es-DlmMVaBG.mjs"), "./translations/fr.json": () => import("./fr-3S6ke71d.mjs"), "./translations/ko.json": () => import("./ko-qTjQ8IMw.mjs"), "./translations/pl.json": () => import("./pl-B67TSHqT.mjs"), "./translations/ru.json": () => import("./ru-hagMa57T.mjs"), "./translations/tr.json": () => import("./tr-Dw_jmkG-.mjs"), "./translations/zh-Hans.json": () => import("./zh-Hans-Dyc-aR-h.mjs"), "./translations/zh.json": () => import("./zh-57YM4amO.mjs") }), `./translations/${locale}.json`).then(({ default: data }) => {
1545
+ return __variableDynamicImportRuntimeHelper(/* @__PURE__ */ Object.assign({ "./translations/de.json": () => import("./de-9eCAqqrB.mjs"), "./translations/dk.json": () => import("./dk-2qBjxt-P.mjs"), "./translations/en.json": () => import("./en-DZXpOMHo.mjs"), "./translations/es.json": () => import("./es-DlmMVaBG.mjs"), "./translations/fr.json": () => import("./fr-3S6ke71d.mjs"), "./translations/ko.json": () => import("./ko-qTjQ8IMw.mjs"), "./translations/pl.json": () => import("./pl-B67TSHqT.mjs"), "./translations/ru.json": () => import("./ru-hagMa57T.mjs"), "./translations/tr.json": () => import("./tr-Dw_jmkG-.mjs"), "./translations/zh-Hans.json": () => import("./zh-Hans-Dyc-aR-h.mjs"), "./translations/zh.json": () => import("./zh-57YM4amO.mjs") }), `./translations/${locale}.json`).then(({ default: data }) => {
1032
1546
  return {
1033
1547
  data: prefixPluginTranslations(data, pluginId),
1034
1548
  locale
@@ -1047,12 +1561,11 @@ const index = {
1047
1561
  export {
1048
1562
  PERMISSIONS as P,
1049
1563
  useGetDefaultLocalesQuery as a,
1050
- useGetLocalesQuery as b,
1051
- useDeleteLocaleMutation as c,
1052
- useUpdateLocaleMutation as d,
1053
- index as e,
1564
+ useDeleteLocaleMutation as b,
1565
+ useUpdateLocaleMutation as c,
1566
+ useGetLocalesQuery as d,
1054
1567
  getTranslation as g,
1055
- isBaseQueryError as i,
1568
+ index as i,
1056
1569
  useCreateLocaleMutation as u
1057
1570
  };
1058
- //# sourceMappingURL=index-lckTPHiZ.mjs.map
1571
+ //# sourceMappingURL=index-DXrgAtCA.mjs.map