@strapi/content-manager 0.0.0-experimental.25e22c6cc9bc6b35392bb55d09f641a0a65e7403 → 0.0.0-experimental.2cfaca2410c03f1dee31ca18c06aedfb313e0fb4
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.
- package/LICENSE +18 -3
- package/dist/_chunks/{ComponentConfigurationPage-WRPUXGd6.js → ComponentConfigurationPage-DnnZJc1F.js} +3 -3
- package/dist/_chunks/{ComponentConfigurationPage-WRPUXGd6.js.map → ComponentConfigurationPage-DnnZJc1F.js.map} +1 -1
- package/dist/_chunks/{ComponentConfigurationPage-gdUj_t-O.mjs → ComponentConfigurationPage-hLMNf7KI.mjs} +3 -3
- package/dist/_chunks/{ComponentConfigurationPage-gdUj_t-O.mjs.map → ComponentConfigurationPage-hLMNf7KI.mjs.map} +1 -1
- package/dist/_chunks/{EditConfigurationPage-C1vjMBgy.js → EditConfigurationPage-CpLj5gYZ.js} +3 -3
- package/dist/_chunks/{EditConfigurationPage-C1vjMBgy.js.map → EditConfigurationPage-CpLj5gYZ.js.map} +1 -1
- package/dist/_chunks/{EditConfigurationPage-BwuIPOJG.mjs → EditConfigurationPage-Dh6sq-G4.mjs} +3 -3
- package/dist/_chunks/{EditConfigurationPage-BwuIPOJG.mjs.map → EditConfigurationPage-Dh6sq-G4.mjs.map} +1 -1
- package/dist/_chunks/{EditViewPage-0MiFkXa8.mjs → EditViewPage-BU1ugeVi.mjs} +19 -8
- package/dist/_chunks/EditViewPage-BU1ugeVi.mjs.map +1 -0
- package/dist/_chunks/{EditViewPage-DbcGfyqK.js → EditViewPage-D2QVRr_2.js} +19 -8
- package/dist/_chunks/EditViewPage-D2QVRr_2.js.map +1 -0
- package/dist/_chunks/{Field-BG1xu38N.js → Field-BEDX9i_V.js} +465 -145
- package/dist/_chunks/Field-BEDX9i_V.js.map +1 -0
- package/dist/_chunks/{Field-BDMSCcy5.mjs → Field-VSPY6uzs.mjs} +463 -143
- package/dist/_chunks/Field-VSPY6uzs.mjs.map +1 -0
- package/dist/_chunks/{Form-CPVWavB8.mjs → Form-05Oaes1N.mjs} +39 -17
- package/dist/_chunks/Form-05Oaes1N.mjs.map +1 -0
- package/dist/_chunks/{Form-9BnFyUjy.js → Form-DCaY8xBX.js} +39 -17
- package/dist/_chunks/Form-DCaY8xBX.js.map +1 -0
- package/dist/_chunks/{History-BVpd8LP3.mjs → History-BqO2G3MV.mjs} +44 -19
- package/dist/_chunks/History-BqO2G3MV.mjs.map +1 -0
- package/dist/_chunks/{History-BWWxLt2Z.js → History-BrJ1tUvt.js} +44 -19
- package/dist/_chunks/History-BrJ1tUvt.js.map +1 -0
- package/dist/_chunks/{ListConfigurationPage-DozVMKcR.mjs → ListConfigurationPage-C6rsFlme.mjs} +20 -8
- package/dist/_chunks/ListConfigurationPage-C6rsFlme.mjs.map +1 -0
- package/dist/_chunks/{ListConfigurationPage-6swzjdAZ.js → ListConfigurationPage-Eane5LKE.js} +20 -8
- package/dist/_chunks/ListConfigurationPage-Eane5LKE.js.map +1 -0
- package/dist/_chunks/{ListViewPage-BlzfjS2Q.js → ListViewPage-Coj-RPsx.js} +61 -43
- package/dist/_chunks/ListViewPage-Coj-RPsx.js.map +1 -0
- package/dist/_chunks/{ListViewPage-Ds0ulgfG.mjs → ListViewPage-yE_zYhcI.mjs} +59 -41
- package/dist/_chunks/ListViewPage-yE_zYhcI.mjs.map +1 -0
- package/dist/_chunks/{NoContentTypePage-D2nCCWEl.js → NoContentTypePage-BDJ0dshy.js} +2 -2
- package/dist/_chunks/{NoContentTypePage-D2nCCWEl.js.map → NoContentTypePage-BDJ0dshy.js.map} +1 -1
- package/dist/_chunks/{NoContentTypePage-BH11kaKt.mjs → NoContentTypePage-NW_FSVdY.mjs} +2 -2
- package/dist/_chunks/{NoContentTypePage-BH11kaKt.mjs.map → NoContentTypePage-NW_FSVdY.mjs.map} +1 -1
- package/dist/_chunks/{NoPermissionsPage-DN_JlsU2.js → NoPermissionsPage-BOtb5FTM.js} +2 -2
- package/dist/_chunks/{NoPermissionsPage-DN_JlsU2.js.map → NoPermissionsPage-BOtb5FTM.js.map} +1 -1
- package/dist/_chunks/{NoPermissionsPage-BT2Tn0D_.mjs → NoPermissionsPage-h0I3ImsX.mjs} +2 -2
- package/dist/_chunks/{NoPermissionsPage-BT2Tn0D_.mjs.map → NoPermissionsPage-h0I3ImsX.mjs.map} +1 -1
- package/dist/_chunks/{Relations-CcgFTcWo.js → Relations-CVh0DOKv.js} +4 -4
- package/dist/_chunks/Relations-CVh0DOKv.js.map +1 -0
- package/dist/_chunks/{Relations-Dnag3fhV.mjs → Relations-FP0uWpBz.mjs} +4 -4
- package/dist/_chunks/Relations-FP0uWpBz.mjs.map +1 -0
- package/dist/_chunks/{en-fbKQxLGn.js → en-BlhnxQfj.js} +11 -9
- package/dist/_chunks/{en-fbKQxLGn.js.map → en-BlhnxQfj.js.map} +1 -1
- package/dist/_chunks/{en-Ux26r5pl.mjs → en-C8YBvRrK.mjs} +11 -9
- package/dist/_chunks/{en-Ux26r5pl.mjs.map → en-C8YBvRrK.mjs.map} +1 -1
- package/dist/_chunks/{index-JNNNKUHs.mjs → index-CPCHQ3X_.mjs} +976 -658
- package/dist/_chunks/index-CPCHQ3X_.mjs.map +1 -0
- package/dist/_chunks/{index-CWpLBSt0.js → index-DTKVhcla.js} +968 -650
- package/dist/_chunks/index-DTKVhcla.js.map +1 -0
- package/dist/_chunks/{layout-DC503LnF.mjs → layout-B4UhJ8MJ.mjs} +27 -14
- package/dist/_chunks/layout-B4UhJ8MJ.mjs.map +1 -0
- package/dist/_chunks/{layout--iHdZzRk.js → layout-CWgZzMYf.js} +25 -12
- package/dist/_chunks/layout-CWgZzMYf.js.map +1 -0
- package/dist/_chunks/{relations-CTje5t-a.mjs → relations-B83Ge9a7.mjs} +2 -2
- package/dist/_chunks/{relations-CTje5t-a.mjs.map → relations-B83Ge9a7.mjs.map} +1 -1
- package/dist/_chunks/{relations-BbHizA5K.js → relations-D81a_2zw.js} +2 -2
- package/dist/_chunks/{relations-BbHizA5K.js.map → relations-D81a_2zw.js.map} +1 -1
- package/dist/_chunks/{usePrev-B9w_-eYc.js → useDebounce-CtcjDB3L.js} +14 -1
- package/dist/_chunks/useDebounce-CtcjDB3L.js.map +1 -0
- package/dist/_chunks/useDebounce-DmuSJIF3.mjs +29 -0
- package/dist/_chunks/useDebounce-DmuSJIF3.mjs.map +1 -0
- package/dist/admin/index.js +2 -1
- package/dist/admin/index.js.map +1 -1
- package/dist/admin/index.mjs +3 -2
- package/dist/admin/src/exports.d.ts +1 -1
- package/dist/admin/src/history/index.d.ts +3 -0
- package/dist/admin/src/history/services/historyVersion.d.ts +1 -1
- package/dist/admin/src/hooks/useDocument.d.ts +30 -1
- package/dist/admin/src/index.d.ts +1 -0
- package/dist/admin/src/pages/EditView/components/DocumentActions.d.ts +1 -0
- package/dist/admin/src/pages/EditView/components/FormInputs/BlocksInput/utils/constants.d.ts +4 -0
- package/dist/admin/src/pages/EditView/components/FormInputs/Relations.d.ts +20 -0
- package/dist/admin/src/pages/EditView/components/FormInputs/Wysiwyg/EditorLayout.d.ts +2 -2
- package/dist/admin/src/pages/EditView/components/FormInputs/Wysiwyg/WysiwygFooter.d.ts +2 -2
- package/dist/admin/src/pages/EditView/components/FormInputs/Wysiwyg/WysiwygStyles.d.ts +4 -48
- package/dist/admin/src/pages/EditView/components/Header.d.ts +10 -11
- package/dist/admin/src/services/api.d.ts +1 -1
- package/dist/admin/src/services/components.d.ts +2 -2
- package/dist/admin/src/services/contentTypes.d.ts +3 -3
- package/dist/admin/src/services/documents.d.ts +19 -17
- package/dist/admin/src/services/init.d.ts +1 -1
- package/dist/admin/src/services/relations.d.ts +2 -2
- package/dist/admin/src/services/uid.d.ts +3 -3
- package/dist/admin/src/utils/validation.d.ts +4 -1
- package/dist/server/index.js +185 -113
- package/dist/server/index.js.map +1 -1
- package/dist/server/index.mjs +186 -114
- package/dist/server/index.mjs.map +1 -1
- package/dist/server/src/controllers/collection-types.d.ts.map +1 -1
- package/dist/server/src/controllers/relations.d.ts.map +1 -1
- package/dist/server/src/controllers/uid.d.ts.map +1 -1
- package/dist/server/src/controllers/validation/dimensions.d.ts +4 -2
- package/dist/server/src/controllers/validation/dimensions.d.ts.map +1 -1
- package/dist/server/src/history/services/history.d.ts.map +1 -1
- package/dist/server/src/history/services/lifecycles.d.ts.map +1 -1
- package/dist/server/src/history/services/utils.d.ts +2 -1
- package/dist/server/src/history/services/utils.d.ts.map +1 -1
- package/dist/server/src/policies/hasPermissions.d.ts.map +1 -1
- package/dist/server/src/services/document-manager.d.ts.map +1 -1
- package/dist/server/src/services/document-metadata.d.ts.map +1 -1
- package/dist/server/src/services/permission-checker.d.ts.map +1 -1
- package/dist/server/src/services/utils/populate.d.ts.map +1 -1
- package/dist/shared/contracts/collection-types.d.ts +3 -1
- package/dist/shared/contracts/collection-types.d.ts.map +1 -1
- package/package.json +11 -11
- package/dist/_chunks/EditViewPage-0MiFkXa8.mjs.map +0 -1
- package/dist/_chunks/EditViewPage-DbcGfyqK.js.map +0 -1
- package/dist/_chunks/Field-BDMSCcy5.mjs.map +0 -1
- package/dist/_chunks/Field-BG1xu38N.js.map +0 -1
- package/dist/_chunks/Form-9BnFyUjy.js.map +0 -1
- package/dist/_chunks/Form-CPVWavB8.mjs.map +0 -1
- package/dist/_chunks/History-BVpd8LP3.mjs.map +0 -1
- package/dist/_chunks/History-BWWxLt2Z.js.map +0 -1
- package/dist/_chunks/ListConfigurationPage-6swzjdAZ.js.map +0 -1
- package/dist/_chunks/ListConfigurationPage-DozVMKcR.mjs.map +0 -1
- package/dist/_chunks/ListViewPage-BlzfjS2Q.js.map +0 -1
- package/dist/_chunks/ListViewPage-Ds0ulgfG.mjs.map +0 -1
- package/dist/_chunks/Relations-CcgFTcWo.js.map +0 -1
- package/dist/_chunks/Relations-Dnag3fhV.mjs.map +0 -1
- package/dist/_chunks/index-CWpLBSt0.js.map +0 -1
- package/dist/_chunks/index-JNNNKUHs.mjs.map +0 -1
- package/dist/_chunks/layout--iHdZzRk.js.map +0 -1
- package/dist/_chunks/layout-DC503LnF.mjs.map +0 -1
- package/dist/_chunks/usePrev-B9w_-eYc.js.map +0 -1
- package/dist/_chunks/usePrev-DH6iah0A.mjs +0 -16
- package/dist/_chunks/usePrev-DH6iah0A.mjs.map +0 -1
- package/strapi-server.js +0 -3
@@ -1,17 +1,17 @@
|
|
1
|
-
import {
|
1
|
+
import { More, Cross, WarningCircle, ListPlus, Pencil, Trash, Check, CrossCircle, CheckCircle, ArrowsCounterClockwise, ChevronRight, Duplicate, ClockCounterClockwise, Feather } from "@strapi/icons";
|
2
2
|
import { jsx, Fragment, jsxs } from "react/jsx-runtime";
|
3
|
-
import { useStrapiApp,
|
4
|
-
import { stringify } from "qs";
|
5
|
-
import { useIntl } from "react-intl";
|
6
|
-
import { useNavigate, useParams, Navigate, useMatch, useLocation, Link, NavLink } from "react-router-dom";
|
3
|
+
import { useStrapiApp, createContext, useAuth, useRBAC, Page, adminApi, translatedErrors, useNotification, useAPIErrorHandler, useQueryParams, getYupValidationErrors, useForm, useTracking, useGuidedTour, BackButton, DescriptionComponentRenderer, useTable, Table } from "@strapi/admin/strapi-admin";
|
7
4
|
import * as React from "react";
|
8
5
|
import { lazy } from "react";
|
9
|
-
import { Button, Menu, VisuallyHidden, Flex, Typography, Dialog, Modal, Radio, Status, SingleSelect, SingleSelectOption,
|
10
|
-
import {
|
6
|
+
import { Button, Menu, VisuallyHidden, Flex, Typography, Dialog, Modal, Radio, Status, Box, SingleSelect, SingleSelectOption, IconButton, Loader, Tooltip, LinkButton } from "@strapi/design-system";
|
7
|
+
import { useIntl } from "react-intl";
|
8
|
+
import { useParams, useNavigate, Navigate, useMatch, useLocation, Link, NavLink } from "react-router-dom";
|
11
9
|
import * as yup from "yup";
|
12
10
|
import { ValidationError } from "yup";
|
13
11
|
import pipe from "lodash/fp/pipe";
|
14
12
|
import { intervalToDuration, isPast } from "date-fns";
|
13
|
+
import { styled } from "styled-components";
|
14
|
+
import { stringify } from "qs";
|
15
15
|
import { createSlice, combineReducers } from "@reduxjs/toolkit";
|
16
16
|
const __variableDynamicImportRuntimeHelper = (glob, path) => {
|
17
17
|
const v = glob[path];
|
@@ -49,42 +49,6 @@ const useInjectionZone = (area) => {
|
|
49
49
|
const [page, position] = area.split(".");
|
50
50
|
return contentManagerPlugin.getInjectedComponents(page, position);
|
51
51
|
};
|
52
|
-
const HistoryAction = ({ model, document }) => {
|
53
|
-
const { formatMessage } = useIntl();
|
54
|
-
const [{ query }] = useQueryParams();
|
55
|
-
const navigate = useNavigate();
|
56
|
-
const pluginsQueryParams = stringify({ plugins: query.plugins }, { encode: false });
|
57
|
-
if (!window.strapi.features.isEnabled("cms-content-history")) {
|
58
|
-
return null;
|
59
|
-
}
|
60
|
-
return {
|
61
|
-
icon: /* @__PURE__ */ jsx(ClockCounterClockwise, {}),
|
62
|
-
label: formatMessage({
|
63
|
-
id: "content-manager.history.document-action",
|
64
|
-
defaultMessage: "Content History"
|
65
|
-
}),
|
66
|
-
onClick: () => navigate({ pathname: "history", search: pluginsQueryParams }),
|
67
|
-
disabled: (
|
68
|
-
/**
|
69
|
-
* The user is creating a new document.
|
70
|
-
* It hasn't been saved yet, so there's no history to go to
|
71
|
-
*/
|
72
|
-
!document || /**
|
73
|
-
* The document has been created but the current dimension has never been saved.
|
74
|
-
* For example, the user is creating a new locale in an existing document,
|
75
|
-
* so there's no history for the document in that locale
|
76
|
-
*/
|
77
|
-
!document.id || /**
|
78
|
-
* History is only available for content types created by the user.
|
79
|
-
* These have the `api::` prefix, as opposed to the ones created by Strapi or plugins,
|
80
|
-
* which start with `admin::` or `plugin::`
|
81
|
-
*/
|
82
|
-
!model.startsWith("api::")
|
83
|
-
),
|
84
|
-
position: "header"
|
85
|
-
};
|
86
|
-
};
|
87
|
-
HistoryAction.type = "history";
|
88
52
|
const ID = "id";
|
89
53
|
const CREATED_BY_ATTRIBUTE_NAME = "createdBy";
|
90
54
|
const UPDATED_BY_ATTRIBUTE_NAME = "updatedBy";
|
@@ -194,10 +158,12 @@ const contentManagerApi = adminApi.enhanceEndpoints({
|
|
194
158
|
"Document",
|
195
159
|
"InitialData",
|
196
160
|
"HistoryVersion",
|
197
|
-
"Relations"
|
161
|
+
"Relations",
|
162
|
+
"UidAvailability"
|
198
163
|
]
|
199
164
|
});
|
200
165
|
const documentApi = contentManagerApi.injectEndpoints({
|
166
|
+
overrideExisting: true,
|
201
167
|
endpoints: (builder) => ({
|
202
168
|
autoCloneDocument: builder.mutation({
|
203
169
|
query: ({ model, sourceId, query }) => ({
|
@@ -207,7 +173,12 @@ const documentApi = contentManagerApi.injectEndpoints({
|
|
207
173
|
params: query
|
208
174
|
}
|
209
175
|
}),
|
210
|
-
invalidatesTags: (_result,
|
176
|
+
invalidatesTags: (_result, error, { model }) => {
|
177
|
+
if (error) {
|
178
|
+
return [];
|
179
|
+
}
|
180
|
+
return [{ type: "Document", id: `${model}_LIST` }];
|
181
|
+
}
|
211
182
|
}),
|
212
183
|
cloneDocument: builder.mutation({
|
213
184
|
query: ({ model, sourceId, data, params }) => ({
|
@@ -218,7 +189,10 @@ const documentApi = contentManagerApi.injectEndpoints({
|
|
218
189
|
params
|
219
190
|
}
|
220
191
|
}),
|
221
|
-
invalidatesTags: (_result, _error, { model }) => [
|
192
|
+
invalidatesTags: (_result, _error, { model }) => [
|
193
|
+
{ type: "Document", id: `${model}_LIST` },
|
194
|
+
{ type: "UidAvailability", id: model }
|
195
|
+
]
|
222
196
|
}),
|
223
197
|
/**
|
224
198
|
* Creates a new collection-type document. This should ONLY be used for collection-types.
|
@@ -235,7 +209,8 @@ const documentApi = contentManagerApi.injectEndpoints({
|
|
235
209
|
}),
|
236
210
|
invalidatesTags: (result, _error, { model }) => [
|
237
211
|
{ type: "Document", id: `${model}_LIST` },
|
238
|
-
"Relations"
|
212
|
+
"Relations",
|
213
|
+
{ type: "UidAvailability", id: model }
|
239
214
|
]
|
240
215
|
}),
|
241
216
|
deleteDocument: builder.mutation({
|
@@ -276,7 +251,8 @@ const documentApi = contentManagerApi.injectEndpoints({
|
|
276
251
|
id: collectionType !== SINGLE_TYPES ? `${model}_${documentId}` : model
|
277
252
|
},
|
278
253
|
{ type: "Document", id: `${model}_LIST` },
|
279
|
-
"Relations"
|
254
|
+
"Relations",
|
255
|
+
{ type: "UidAvailability", id: model }
|
280
256
|
];
|
281
257
|
}
|
282
258
|
}),
|
@@ -294,6 +270,7 @@ const documentApi = contentManagerApi.injectEndpoints({
|
|
294
270
|
}),
|
295
271
|
providesTags: (result, _error, arg) => {
|
296
272
|
return [
|
273
|
+
{ type: "Document", id: `ALL_LIST` },
|
297
274
|
{ type: "Document", id: `${arg.model}_LIST` },
|
298
275
|
...result?.results.map(({ documentId }) => ({
|
299
276
|
type: "Document",
|
@@ -332,6 +309,11 @@ const documentApi = contentManagerApi.injectEndpoints({
|
|
332
309
|
{
|
333
310
|
type: "Document",
|
334
311
|
id: collectionType !== SINGLE_TYPES ? `${model}_${result && "documentId" in result ? result.documentId : documentId}` : model
|
312
|
+
},
|
313
|
+
// Make it easy to invalidate all individual documents queries for a model
|
314
|
+
{
|
315
|
+
type: "Document",
|
316
|
+
id: `${model}_ALL_ITEMS`
|
335
317
|
}
|
336
318
|
];
|
337
319
|
}
|
@@ -395,8 +377,21 @@ const documentApi = contentManagerApi.injectEndpoints({
|
|
395
377
|
type: "Document",
|
396
378
|
id: collectionType !== SINGLE_TYPES ? `${model}_${documentId}` : model
|
397
379
|
},
|
398
|
-
"Relations"
|
380
|
+
"Relations",
|
381
|
+
{ type: "UidAvailability", id: model }
|
399
382
|
];
|
383
|
+
},
|
384
|
+
async onQueryStarted({ data, ...patch }, { dispatch, queryFulfilled }) {
|
385
|
+
const patchResult = dispatch(
|
386
|
+
documentApi.util.updateQueryData("getDocument", patch, (draft) => {
|
387
|
+
Object.assign(draft.data, data);
|
388
|
+
})
|
389
|
+
);
|
390
|
+
try {
|
391
|
+
await queryFulfilled;
|
392
|
+
} catch {
|
393
|
+
patchResult.undo();
|
394
|
+
}
|
400
395
|
}
|
401
396
|
}),
|
402
397
|
unpublishDocument: builder.mutation({
|
@@ -466,7 +461,7 @@ const buildValidParams = (query) => {
|
|
466
461
|
const isBaseQueryError = (error) => {
|
467
462
|
return error.name !== void 0;
|
468
463
|
};
|
469
|
-
const createYupSchema = (attributes = {}, components = {}) => {
|
464
|
+
const createYupSchema = (attributes = {}, components = {}, options = { status: null }) => {
|
470
465
|
const createModelSchema = (attributes2) => yup.object().shape(
|
471
466
|
Object.entries(attributes2).reduce((acc, [name, attribute]) => {
|
472
467
|
if (DOCUMENT_META_FIELDS.includes(name)) {
|
@@ -479,7 +474,7 @@ const createYupSchema = (attributes = {}, components = {}) => {
|
|
479
474
|
addMinValidation,
|
480
475
|
addMaxValidation,
|
481
476
|
addRegexValidation
|
482
|
-
].map((fn) => fn(attribute));
|
477
|
+
].map((fn) => fn(attribute, options));
|
483
478
|
const transformSchema = pipe(...validations);
|
484
479
|
switch (attribute.type) {
|
485
480
|
case "component": {
|
@@ -580,6 +575,14 @@ const createAttributeSchema = (attribute) => {
|
|
580
575
|
if (!value || typeof value === "string" && value.length === 0) {
|
581
576
|
return true;
|
582
577
|
}
|
578
|
+
if (typeof value === "object") {
|
579
|
+
try {
|
580
|
+
JSON.stringify(value);
|
581
|
+
return true;
|
582
|
+
} catch (err) {
|
583
|
+
return false;
|
584
|
+
}
|
585
|
+
}
|
583
586
|
try {
|
584
587
|
JSON.parse(value);
|
585
588
|
return true;
|
@@ -598,13 +601,7 @@ const createAttributeSchema = (attribute) => {
|
|
598
601
|
return yup.mixed();
|
599
602
|
}
|
600
603
|
};
|
601
|
-
const
|
602
|
-
if (attribute.required) {
|
603
|
-
return schema.required({
|
604
|
-
id: translatedErrors.required.id,
|
605
|
-
defaultMessage: "This field is required."
|
606
|
-
});
|
607
|
-
}
|
604
|
+
const nullableSchema = (schema) => {
|
608
605
|
return schema?.nullable ? schema.nullable() : (
|
609
606
|
// In some cases '.nullable' will not be available on the schema.
|
610
607
|
// e.g. when the schema has been built using yup.lazy (e.g. for relations).
|
@@ -612,7 +609,22 @@ const addRequiredValidation = (attribute) => (schema) => {
|
|
612
609
|
schema
|
613
610
|
);
|
614
611
|
};
|
615
|
-
const
|
612
|
+
const addRequiredValidation = (attribute, options) => (schema) => {
|
613
|
+
if (options.status === "draft") {
|
614
|
+
return nullableSchema(schema);
|
615
|
+
}
|
616
|
+
if ((attribute.type === "component" && attribute.repeatable || attribute.type === "dynamiczone") && attribute.required && "min" in schema) {
|
617
|
+
return schema.min(1, translatedErrors.required);
|
618
|
+
}
|
619
|
+
if (attribute.required && attribute.type !== "relation") {
|
620
|
+
return schema.required(translatedErrors.required);
|
621
|
+
}
|
622
|
+
return nullableSchema(schema);
|
623
|
+
};
|
624
|
+
const addMinLengthValidation = (attribute, options) => (schema) => {
|
625
|
+
if (options.status === "draft") {
|
626
|
+
return schema;
|
627
|
+
}
|
616
628
|
if ("minLength" in attribute && attribute.minLength && Number.isInteger(attribute.minLength) && "min" in schema) {
|
617
629
|
return schema.min(attribute.minLength, {
|
618
630
|
...translatedErrors.minLength,
|
@@ -634,9 +646,31 @@ const addMaxLengthValidation = (attribute) => (schema) => {
|
|
634
646
|
}
|
635
647
|
return schema;
|
636
648
|
};
|
637
|
-
const addMinValidation = (attribute) => (schema) => {
|
649
|
+
const addMinValidation = (attribute, options) => (schema) => {
|
638
650
|
if ("min" in attribute) {
|
639
651
|
const min = toInteger(attribute.min);
|
652
|
+
if (attribute.type === "component" && attribute.repeatable || attribute.type === "dynamiczone") {
|
653
|
+
if (options.status !== "draft" && !attribute.required && "test" in schema && min) {
|
654
|
+
return schema.test(
|
655
|
+
"custom-min",
|
656
|
+
{
|
657
|
+
...translatedErrors.min,
|
658
|
+
values: {
|
659
|
+
min: attribute.min
|
660
|
+
}
|
661
|
+
},
|
662
|
+
(value) => {
|
663
|
+
if (!value) {
|
664
|
+
return true;
|
665
|
+
}
|
666
|
+
if (Array.isArray(value) && value.length === 0) {
|
667
|
+
return true;
|
668
|
+
}
|
669
|
+
return value.length >= min;
|
670
|
+
}
|
671
|
+
);
|
672
|
+
}
|
673
|
+
}
|
640
674
|
if ("min" in schema && min) {
|
641
675
|
return schema.min(min, {
|
642
676
|
...translatedErrors.min,
|
@@ -755,16 +789,115 @@ const extractContentTypeComponents = (attributes = {}, allComponents = {}) => {
|
|
755
789
|
}, {});
|
756
790
|
return componentsByKey;
|
757
791
|
};
|
758
|
-
const
|
792
|
+
const HOOKS = {
|
793
|
+
/**
|
794
|
+
* Hook that allows to mutate the displayed headers of the list view table
|
795
|
+
* @constant
|
796
|
+
* @type {string}
|
797
|
+
*/
|
798
|
+
INJECT_COLUMN_IN_TABLE: "Admin/CM/pages/ListView/inject-column-in-table",
|
799
|
+
/**
|
800
|
+
* Hook that allows to mutate the CM's collection types links pre-set filters
|
801
|
+
* @constant
|
802
|
+
* @type {string}
|
803
|
+
*/
|
804
|
+
MUTATE_COLLECTION_TYPES_LINKS: "Admin/CM/pages/App/mutate-collection-types-links",
|
805
|
+
/**
|
806
|
+
* Hook that allows to mutate the CM's edit view layout
|
807
|
+
* @constant
|
808
|
+
* @type {string}
|
809
|
+
*/
|
810
|
+
MUTATE_EDIT_VIEW_LAYOUT: "Admin/CM/pages/EditView/mutate-edit-view-layout",
|
811
|
+
/**
|
812
|
+
* Hook that allows to mutate the CM's single types links pre-set filters
|
813
|
+
* @constant
|
814
|
+
* @type {string}
|
815
|
+
*/
|
816
|
+
MUTATE_SINGLE_TYPES_LINKS: "Admin/CM/pages/App/mutate-single-types-links"
|
817
|
+
};
|
818
|
+
const contentTypesApi = contentManagerApi.injectEndpoints({
|
819
|
+
endpoints: (builder) => ({
|
820
|
+
getContentTypeConfiguration: builder.query({
|
821
|
+
query: (uid) => ({
|
822
|
+
url: `/content-manager/content-types/${uid}/configuration`,
|
823
|
+
method: "GET"
|
824
|
+
}),
|
825
|
+
transformResponse: (response) => response.data,
|
826
|
+
providesTags: (_result, _error, uid) => [
|
827
|
+
{ type: "ContentTypesConfiguration", id: uid },
|
828
|
+
{ type: "ContentTypeSettings", id: "LIST" }
|
829
|
+
]
|
830
|
+
}),
|
831
|
+
getAllContentTypeSettings: builder.query({
|
832
|
+
query: () => "/content-manager/content-types-settings",
|
833
|
+
transformResponse: (response) => response.data,
|
834
|
+
providesTags: [{ type: "ContentTypeSettings", id: "LIST" }]
|
835
|
+
}),
|
836
|
+
updateContentTypeConfiguration: builder.mutation({
|
837
|
+
query: ({ uid, ...body }) => ({
|
838
|
+
url: `/content-manager/content-types/${uid}/configuration`,
|
839
|
+
method: "PUT",
|
840
|
+
data: body
|
841
|
+
}),
|
842
|
+
transformResponse: (response) => response.data,
|
843
|
+
invalidatesTags: (_result, _error, { uid }) => [
|
844
|
+
{ type: "ContentTypesConfiguration", id: uid },
|
845
|
+
{ type: "ContentTypeSettings", id: "LIST" },
|
846
|
+
// Is this necessary?
|
847
|
+
{ type: "InitialData" }
|
848
|
+
]
|
849
|
+
})
|
850
|
+
})
|
851
|
+
});
|
852
|
+
const {
|
853
|
+
useGetContentTypeConfigurationQuery,
|
854
|
+
useGetAllContentTypeSettingsQuery,
|
855
|
+
useUpdateContentTypeConfigurationMutation
|
856
|
+
} = contentTypesApi;
|
857
|
+
const checkIfAttributeIsDisplayable = (attribute) => {
|
858
|
+
const { type } = attribute;
|
859
|
+
if (type === "relation") {
|
860
|
+
return !attribute.relation.toLowerCase().includes("morph");
|
861
|
+
}
|
862
|
+
return !["json", "dynamiczone", "richtext", "password", "blocks"].includes(type) && !!type;
|
863
|
+
};
|
864
|
+
const getMainField = (attribute, mainFieldName, { schemas, components }) => {
|
865
|
+
if (!mainFieldName) {
|
866
|
+
return void 0;
|
867
|
+
}
|
868
|
+
const mainFieldType = attribute.type === "component" ? components[attribute.component].attributes[mainFieldName].type : (
|
869
|
+
// @ts-expect-error – `targetModel` does exist on the attribute for a relation.
|
870
|
+
schemas.find((schema) => schema.uid === attribute.targetModel)?.attributes[mainFieldName].type
|
871
|
+
);
|
872
|
+
return {
|
873
|
+
name: mainFieldName,
|
874
|
+
type: mainFieldType ?? "string"
|
875
|
+
};
|
876
|
+
};
|
877
|
+
const DEFAULT_SETTINGS = {
|
878
|
+
bulkable: false,
|
879
|
+
filterable: false,
|
880
|
+
searchable: false,
|
881
|
+
pagination: false,
|
882
|
+
defaultSortBy: "",
|
883
|
+
defaultSortOrder: "asc",
|
884
|
+
mainField: "id",
|
885
|
+
pageSize: 10
|
886
|
+
};
|
887
|
+
const useDocumentLayout = (model) => {
|
888
|
+
const { schema, components } = useDocument({ model, collectionType: "" }, { skip: true });
|
889
|
+
const [{ query }] = useQueryParams();
|
890
|
+
const runHookWaterfall = useStrapiApp("useDocumentLayout", (state) => state.runHookWaterfall);
|
759
891
|
const { toggleNotification } = useNotification();
|
760
892
|
const { _unstableFormatAPIError: formatAPIError } = useAPIErrorHandler();
|
893
|
+
const { isLoading: isLoadingSchemas, schemas } = useContentTypeSchema();
|
761
894
|
const {
|
762
|
-
|
763
|
-
isLoading:
|
764
|
-
|
765
|
-
|
766
|
-
} =
|
767
|
-
const
|
895
|
+
data,
|
896
|
+
isLoading: isLoadingConfigs,
|
897
|
+
error,
|
898
|
+
isFetching: isFetchingConfigs
|
899
|
+
} = useGetContentTypeConfigurationQuery(model);
|
900
|
+
const isLoading = isLoadingSchemas || isFetchingConfigs || isLoadingConfigs;
|
768
901
|
React.useEffect(() => {
|
769
902
|
if (error) {
|
770
903
|
toggleNotification({
|
@@ -772,56 +905,269 @@ const useDocument = (args, opts) => {
|
|
772
905
|
message: formatAPIError(error)
|
773
906
|
});
|
774
907
|
}
|
775
|
-
}, [
|
776
|
-
const
|
777
|
-
|
778
|
-
|
779
|
-
|
780
|
-
|
781
|
-
|
782
|
-
|
783
|
-
(document) => {
|
784
|
-
if (!validationSchema) {
|
785
|
-
throw new Error(
|
786
|
-
"There is no validation schema generated, this is likely due to the schema not being loaded yet."
|
787
|
-
);
|
788
|
-
}
|
789
|
-
try {
|
790
|
-
validationSchema.validateSync(document, { abortEarly: false, strict: true });
|
791
|
-
return null;
|
792
|
-
} catch (error2) {
|
793
|
-
if (error2 instanceof ValidationError) {
|
794
|
-
return getYupValidationErrors(error2);
|
795
|
-
}
|
796
|
-
throw error2;
|
797
|
-
}
|
908
|
+
}, [error, formatAPIError, toggleNotification]);
|
909
|
+
const editLayout = React.useMemo(
|
910
|
+
() => data && !isLoading ? formatEditLayout(data, { schemas, schema, components }) : {
|
911
|
+
layout: [],
|
912
|
+
components: {},
|
913
|
+
metadatas: {},
|
914
|
+
options: {},
|
915
|
+
settings: DEFAULT_SETTINGS
|
798
916
|
},
|
799
|
-
[
|
917
|
+
[data, isLoading, schemas, schema, components]
|
918
|
+
);
|
919
|
+
const listLayout = React.useMemo(() => {
|
920
|
+
return data && !isLoading ? formatListLayout(data, { schemas, schema, components }) : {
|
921
|
+
layout: [],
|
922
|
+
metadatas: {},
|
923
|
+
options: {},
|
924
|
+
settings: DEFAULT_SETTINGS
|
925
|
+
};
|
926
|
+
}, [data, isLoading, schemas, schema, components]);
|
927
|
+
const { layout: edit } = React.useMemo(
|
928
|
+
() => runHookWaterfall(HOOKS.MUTATE_EDIT_VIEW_LAYOUT, {
|
929
|
+
layout: editLayout,
|
930
|
+
query
|
931
|
+
}),
|
932
|
+
[editLayout, query, runHookWaterfall]
|
800
933
|
);
|
801
|
-
const isLoading = isLoadingDocument || isFetchingDocument || isLoadingSchema;
|
802
934
|
return {
|
803
|
-
|
804
|
-
document: data?.data,
|
805
|
-
meta: data?.meta,
|
935
|
+
error,
|
806
936
|
isLoading,
|
807
|
-
|
808
|
-
|
937
|
+
edit,
|
938
|
+
list: listLayout
|
809
939
|
};
|
810
940
|
};
|
811
|
-
const
|
812
|
-
const {
|
813
|
-
|
814
|
-
|
815
|
-
|
816
|
-
|
817
|
-
|
818
|
-
|
819
|
-
|
820
|
-
|
821
|
-
|
822
|
-
|
823
|
-
|
824
|
-
|
941
|
+
const useDocLayout = () => {
|
942
|
+
const { model } = useDoc();
|
943
|
+
return useDocumentLayout(model);
|
944
|
+
};
|
945
|
+
const formatEditLayout = (data, {
|
946
|
+
schemas,
|
947
|
+
schema,
|
948
|
+
components
|
949
|
+
}) => {
|
950
|
+
let currentPanelIndex = 0;
|
951
|
+
const panelledEditAttributes = convertEditLayoutToFieldLayouts(
|
952
|
+
data.contentType.layouts.edit,
|
953
|
+
schema?.attributes,
|
954
|
+
data.contentType.metadatas,
|
955
|
+
{ configurations: data.components, schemas: components },
|
956
|
+
schemas
|
957
|
+
).reduce((panels, row) => {
|
958
|
+
if (row.some((field) => field.type === "dynamiczone")) {
|
959
|
+
panels.push([row]);
|
960
|
+
currentPanelIndex += 2;
|
961
|
+
} else {
|
962
|
+
if (!panels[currentPanelIndex]) {
|
963
|
+
panels.push([]);
|
964
|
+
}
|
965
|
+
panels[currentPanelIndex].push(row);
|
966
|
+
}
|
967
|
+
return panels;
|
968
|
+
}, []);
|
969
|
+
const componentEditAttributes = Object.entries(data.components).reduce(
|
970
|
+
(acc, [uid, configuration]) => {
|
971
|
+
acc[uid] = {
|
972
|
+
layout: convertEditLayoutToFieldLayouts(
|
973
|
+
configuration.layouts.edit,
|
974
|
+
components[uid].attributes,
|
975
|
+
configuration.metadatas,
|
976
|
+
{ configurations: data.components, schemas: components }
|
977
|
+
),
|
978
|
+
settings: {
|
979
|
+
...configuration.settings,
|
980
|
+
icon: components[uid].info.icon,
|
981
|
+
displayName: components[uid].info.displayName
|
982
|
+
}
|
983
|
+
};
|
984
|
+
return acc;
|
985
|
+
},
|
986
|
+
{}
|
987
|
+
);
|
988
|
+
const editMetadatas = Object.entries(data.contentType.metadatas).reduce(
|
989
|
+
(acc, [attribute, metadata]) => {
|
990
|
+
return {
|
991
|
+
...acc,
|
992
|
+
[attribute]: metadata.edit
|
993
|
+
};
|
994
|
+
},
|
995
|
+
{}
|
996
|
+
);
|
997
|
+
return {
|
998
|
+
layout: panelledEditAttributes,
|
999
|
+
components: componentEditAttributes,
|
1000
|
+
metadatas: editMetadatas,
|
1001
|
+
settings: {
|
1002
|
+
...data.contentType.settings,
|
1003
|
+
displayName: schema?.info.displayName
|
1004
|
+
},
|
1005
|
+
options: {
|
1006
|
+
...schema?.options,
|
1007
|
+
...schema?.pluginOptions,
|
1008
|
+
...data.contentType.options
|
1009
|
+
}
|
1010
|
+
};
|
1011
|
+
};
|
1012
|
+
const convertEditLayoutToFieldLayouts = (rows, attributes = {}, metadatas, components, schemas = []) => {
|
1013
|
+
return rows.map(
|
1014
|
+
(row) => row.map((field) => {
|
1015
|
+
const attribute = attributes[field.name];
|
1016
|
+
if (!attribute) {
|
1017
|
+
return null;
|
1018
|
+
}
|
1019
|
+
const { edit: metadata } = metadatas[field.name];
|
1020
|
+
const settings = attribute.type === "component" && components ? components.configurations[attribute.component].settings : {};
|
1021
|
+
return {
|
1022
|
+
attribute,
|
1023
|
+
disabled: !metadata.editable,
|
1024
|
+
hint: metadata.description,
|
1025
|
+
label: metadata.label ?? "",
|
1026
|
+
name: field.name,
|
1027
|
+
// @ts-expect-error – mainField does exist on the metadata for a relation.
|
1028
|
+
mainField: getMainField(attribute, metadata.mainField || settings.mainField, {
|
1029
|
+
schemas,
|
1030
|
+
components: components?.schemas ?? {}
|
1031
|
+
}),
|
1032
|
+
placeholder: metadata.placeholder ?? "",
|
1033
|
+
required: attribute.required ?? false,
|
1034
|
+
size: field.size,
|
1035
|
+
unique: "unique" in attribute ? attribute.unique : false,
|
1036
|
+
visible: metadata.visible ?? true,
|
1037
|
+
type: attribute.type
|
1038
|
+
};
|
1039
|
+
}).filter((field) => field !== null)
|
1040
|
+
);
|
1041
|
+
};
|
1042
|
+
const formatListLayout = (data, {
|
1043
|
+
schemas,
|
1044
|
+
schema,
|
1045
|
+
components
|
1046
|
+
}) => {
|
1047
|
+
const listMetadatas = Object.entries(data.contentType.metadatas).reduce(
|
1048
|
+
(acc, [attribute, metadata]) => {
|
1049
|
+
return {
|
1050
|
+
...acc,
|
1051
|
+
[attribute]: metadata.list
|
1052
|
+
};
|
1053
|
+
},
|
1054
|
+
{}
|
1055
|
+
);
|
1056
|
+
const listAttributes = convertListLayoutToFieldLayouts(
|
1057
|
+
data.contentType.layouts.list,
|
1058
|
+
schema?.attributes,
|
1059
|
+
listMetadatas,
|
1060
|
+
{ configurations: data.components, schemas: components },
|
1061
|
+
schemas
|
1062
|
+
);
|
1063
|
+
return {
|
1064
|
+
layout: listAttributes,
|
1065
|
+
settings: { ...data.contentType.settings, displayName: schema?.info.displayName },
|
1066
|
+
metadatas: listMetadatas,
|
1067
|
+
options: {
|
1068
|
+
...schema?.options,
|
1069
|
+
...schema?.pluginOptions,
|
1070
|
+
...data.contentType.options
|
1071
|
+
}
|
1072
|
+
};
|
1073
|
+
};
|
1074
|
+
const convertListLayoutToFieldLayouts = (columns, attributes = {}, metadatas, components, schemas = []) => {
|
1075
|
+
return columns.map((name) => {
|
1076
|
+
const attribute = attributes[name];
|
1077
|
+
if (!attribute) {
|
1078
|
+
return null;
|
1079
|
+
}
|
1080
|
+
const metadata = metadatas[name];
|
1081
|
+
const settings = attribute.type === "component" && components ? components.configurations[attribute.component].settings : {};
|
1082
|
+
return {
|
1083
|
+
attribute,
|
1084
|
+
label: metadata.label ?? "",
|
1085
|
+
mainField: getMainField(attribute, metadata.mainField || settings.mainField, {
|
1086
|
+
schemas,
|
1087
|
+
components: components?.schemas ?? {}
|
1088
|
+
}),
|
1089
|
+
name,
|
1090
|
+
searchable: metadata.searchable ?? true,
|
1091
|
+
sortable: metadata.sortable ?? true
|
1092
|
+
};
|
1093
|
+
}).filter((field) => field !== null);
|
1094
|
+
};
|
1095
|
+
const useDocument = (args, opts) => {
|
1096
|
+
const { toggleNotification } = useNotification();
|
1097
|
+
const { _unstableFormatAPIError: formatAPIError } = useAPIErrorHandler();
|
1098
|
+
const {
|
1099
|
+
currentData: data,
|
1100
|
+
isLoading: isLoadingDocument,
|
1101
|
+
isFetching: isFetchingDocument,
|
1102
|
+
error
|
1103
|
+
} = useGetDocumentQuery(args, {
|
1104
|
+
...opts,
|
1105
|
+
skip: !args.documentId && args.collectionType !== SINGLE_TYPES || opts?.skip
|
1106
|
+
});
|
1107
|
+
const {
|
1108
|
+
components,
|
1109
|
+
schema,
|
1110
|
+
schemas,
|
1111
|
+
isLoading: isLoadingSchema
|
1112
|
+
} = useContentTypeSchema(args.model);
|
1113
|
+
React.useEffect(() => {
|
1114
|
+
if (error) {
|
1115
|
+
toggleNotification({
|
1116
|
+
type: "danger",
|
1117
|
+
message: formatAPIError(error)
|
1118
|
+
});
|
1119
|
+
}
|
1120
|
+
}, [toggleNotification, error, formatAPIError, args.collectionType]);
|
1121
|
+
const validationSchema = React.useMemo(() => {
|
1122
|
+
if (!schema) {
|
1123
|
+
return null;
|
1124
|
+
}
|
1125
|
+
return createYupSchema(schema.attributes, components);
|
1126
|
+
}, [schema, components]);
|
1127
|
+
const validate = React.useCallback(
|
1128
|
+
(document) => {
|
1129
|
+
if (!validationSchema) {
|
1130
|
+
throw new Error(
|
1131
|
+
"There is no validation schema generated, this is likely due to the schema not being loaded yet."
|
1132
|
+
);
|
1133
|
+
}
|
1134
|
+
try {
|
1135
|
+
validationSchema.validateSync(document, { abortEarly: false, strict: true });
|
1136
|
+
return null;
|
1137
|
+
} catch (error2) {
|
1138
|
+
if (error2 instanceof ValidationError) {
|
1139
|
+
return getYupValidationErrors(error2);
|
1140
|
+
}
|
1141
|
+
throw error2;
|
1142
|
+
}
|
1143
|
+
},
|
1144
|
+
[validationSchema]
|
1145
|
+
);
|
1146
|
+
const isLoading = isLoadingDocument || isFetchingDocument || isLoadingSchema;
|
1147
|
+
return {
|
1148
|
+
components,
|
1149
|
+
document: data?.data,
|
1150
|
+
meta: data?.meta,
|
1151
|
+
isLoading,
|
1152
|
+
schema,
|
1153
|
+
schemas,
|
1154
|
+
validate
|
1155
|
+
};
|
1156
|
+
};
|
1157
|
+
const useDoc = () => {
|
1158
|
+
const { id, slug, collectionType, origin } = useParams();
|
1159
|
+
const [{ query }] = useQueryParams();
|
1160
|
+
const params = React.useMemo(() => buildValidParams(query), [query]);
|
1161
|
+
if (!collectionType) {
|
1162
|
+
throw new Error("Could not find collectionType in url params");
|
1163
|
+
}
|
1164
|
+
if (!slug) {
|
1165
|
+
throw new Error("Could not find model in url params");
|
1166
|
+
}
|
1167
|
+
return {
|
1168
|
+
collectionType,
|
1169
|
+
model: slug,
|
1170
|
+
id: origin || id === "create" ? void 0 : id,
|
825
1171
|
...useDocument(
|
826
1172
|
{ documentId: origin || id, model: slug, collectionType, params },
|
827
1173
|
{
|
@@ -830,6 +1176,45 @@ const useDoc = () => {
|
|
830
1176
|
)
|
831
1177
|
};
|
832
1178
|
};
|
1179
|
+
const useContentManagerContext = () => {
|
1180
|
+
const {
|
1181
|
+
collectionType,
|
1182
|
+
model,
|
1183
|
+
id,
|
1184
|
+
components,
|
1185
|
+
isLoading: isLoadingDoc,
|
1186
|
+
schema,
|
1187
|
+
schemas
|
1188
|
+
} = useDoc();
|
1189
|
+
const layout = useDocumentLayout(model);
|
1190
|
+
const form = useForm("useContentManagerContext", (state) => state);
|
1191
|
+
const isSingleType = collectionType === SINGLE_TYPES;
|
1192
|
+
const slug = model;
|
1193
|
+
const isCreatingEntry = id === "create";
|
1194
|
+
useContentTypeSchema();
|
1195
|
+
const isLoading = isLoadingDoc || layout.isLoading;
|
1196
|
+
const error = layout.error;
|
1197
|
+
return {
|
1198
|
+
error,
|
1199
|
+
isLoading,
|
1200
|
+
// Base metadata
|
1201
|
+
model,
|
1202
|
+
collectionType,
|
1203
|
+
id,
|
1204
|
+
slug,
|
1205
|
+
isCreatingEntry,
|
1206
|
+
isSingleType,
|
1207
|
+
hasDraftAndPublish: schema?.options?.draftAndPublish ?? false,
|
1208
|
+
// All schema infos
|
1209
|
+
components,
|
1210
|
+
contentType: schema,
|
1211
|
+
contentTypes: schemas,
|
1212
|
+
// Form state
|
1213
|
+
form,
|
1214
|
+
// layout infos
|
1215
|
+
layout
|
1216
|
+
};
|
1217
|
+
};
|
833
1218
|
const prefixPluginTranslations = (trad, pluginId) => {
|
834
1219
|
if (!pluginId) {
|
835
1220
|
throw new TypeError("pluginId can't be empty");
|
@@ -849,6 +1234,8 @@ const useDocumentActions = () => {
|
|
849
1234
|
const { formatMessage } = useIntl();
|
850
1235
|
const { trackUsage } = useTracking();
|
851
1236
|
const { _unstableFormatAPIError: formatAPIError } = useAPIErrorHandler();
|
1237
|
+
const navigate = useNavigate();
|
1238
|
+
const setCurrentStep = useGuidedTour("useDocumentActions", (state) => state.setCurrentStep);
|
852
1239
|
const [deleteDocument] = useDeleteDocumentMutation();
|
853
1240
|
const _delete = React.useCallback(
|
854
1241
|
async ({ collectionType, model, documentId, params }, trackerProperty) => {
|
@@ -1163,6 +1550,7 @@ const useDocumentActions = () => {
|
|
1163
1550
|
defaultMessage: "Saved document"
|
1164
1551
|
})
|
1165
1552
|
});
|
1553
|
+
setCurrentStep("contentManager.success");
|
1166
1554
|
return res.data;
|
1167
1555
|
} catch (err) {
|
1168
1556
|
toggleNotification({
|
@@ -1184,7 +1572,6 @@ const useDocumentActions = () => {
|
|
1184
1572
|
sourceId
|
1185
1573
|
});
|
1186
1574
|
if ("error" in res) {
|
1187
|
-
toggleNotification({ type: "danger", message: formatAPIError(res.error) });
|
1188
1575
|
return { error: res.error };
|
1189
1576
|
}
|
1190
1577
|
toggleNotification({
|
@@ -1203,7 +1590,7 @@ const useDocumentActions = () => {
|
|
1203
1590
|
throw err;
|
1204
1591
|
}
|
1205
1592
|
},
|
1206
|
-
[autoCloneDocument,
|
1593
|
+
[autoCloneDocument, formatMessage, toggleNotification]
|
1207
1594
|
);
|
1208
1595
|
const [cloneDocument] = useCloneDocumentMutation();
|
1209
1596
|
const clone = React.useCallback(
|
@@ -1229,6 +1616,7 @@ const useDocumentActions = () => {
|
|
1229
1616
|
defaultMessage: "Cloned document"
|
1230
1617
|
})
|
1231
1618
|
});
|
1619
|
+
navigate(`../../${res.data.data.documentId}`, { relative: "path" });
|
1232
1620
|
return res.data;
|
1233
1621
|
} catch (err) {
|
1234
1622
|
toggleNotification({
|
@@ -1239,7 +1627,7 @@ const useDocumentActions = () => {
|
|
1239
1627
|
throw err;
|
1240
1628
|
}
|
1241
1629
|
},
|
1242
|
-
[cloneDocument, trackUsage, toggleNotification, formatMessage, formatAPIError]
|
1630
|
+
[cloneDocument, trackUsage, toggleNotification, formatMessage, formatAPIError, navigate]
|
1243
1631
|
);
|
1244
1632
|
const [getDoc] = useLazyGetDocumentQuery();
|
1245
1633
|
const getDocument = React.useCallback(
|
@@ -1265,7 +1653,7 @@ const useDocumentActions = () => {
|
|
1265
1653
|
};
|
1266
1654
|
};
|
1267
1655
|
const ProtectedHistoryPage = lazy(
|
1268
|
-
() => import("./History-
|
1656
|
+
() => import("./History-BqO2G3MV.mjs").then((mod) => ({ default: mod.ProtectedHistoryPage }))
|
1269
1657
|
);
|
1270
1658
|
const routes$1 = [
|
1271
1659
|
{
|
@@ -1278,31 +1666,31 @@ const routes$1 = [
|
|
1278
1666
|
}
|
1279
1667
|
];
|
1280
1668
|
const ProtectedEditViewPage = lazy(
|
1281
|
-
() => import("./EditViewPage-
|
1669
|
+
() => import("./EditViewPage-BU1ugeVi.mjs").then((mod) => ({ default: mod.ProtectedEditViewPage }))
|
1282
1670
|
);
|
1283
1671
|
const ProtectedListViewPage = lazy(
|
1284
|
-
() => import("./ListViewPage-
|
1672
|
+
() => import("./ListViewPage-yE_zYhcI.mjs").then((mod) => ({ default: mod.ProtectedListViewPage }))
|
1285
1673
|
);
|
1286
1674
|
const ProtectedListConfiguration = lazy(
|
1287
|
-
() => import("./ListConfigurationPage-
|
1675
|
+
() => import("./ListConfigurationPage-C6rsFlme.mjs").then((mod) => ({
|
1288
1676
|
default: mod.ProtectedListConfiguration
|
1289
1677
|
}))
|
1290
1678
|
);
|
1291
1679
|
const ProtectedEditConfigurationPage = lazy(
|
1292
|
-
() => import("./EditConfigurationPage-
|
1680
|
+
() => import("./EditConfigurationPage-Dh6sq-G4.mjs").then((mod) => ({
|
1293
1681
|
default: mod.ProtectedEditConfigurationPage
|
1294
1682
|
}))
|
1295
1683
|
);
|
1296
1684
|
const ProtectedComponentConfigurationPage = lazy(
|
1297
|
-
() => import("./ComponentConfigurationPage-
|
1685
|
+
() => import("./ComponentConfigurationPage-hLMNf7KI.mjs").then((mod) => ({
|
1298
1686
|
default: mod.ProtectedComponentConfigurationPage
|
1299
1687
|
}))
|
1300
1688
|
);
|
1301
1689
|
const NoPermissions = lazy(
|
1302
|
-
() => import("./NoPermissionsPage-
|
1690
|
+
() => import("./NoPermissionsPage-h0I3ImsX.mjs").then((mod) => ({ default: mod.NoPermissions }))
|
1303
1691
|
);
|
1304
1692
|
const NoContentType = lazy(
|
1305
|
-
() => import("./NoContentTypePage-
|
1693
|
+
() => import("./NoContentTypePage-NW_FSVdY.mjs").then((mod) => ({ default: mod.NoContentType }))
|
1306
1694
|
);
|
1307
1695
|
const CollectionTypePages = () => {
|
1308
1696
|
const { collectionType } = useParams();
|
@@ -1416,12 +1804,14 @@ const DocumentActionButton = (action) => {
|
|
1416
1804
|
/* @__PURE__ */ jsx(
|
1417
1805
|
Button,
|
1418
1806
|
{
|
1419
|
-
flex:
|
1807
|
+
flex: "auto",
|
1420
1808
|
startIcon: action.icon,
|
1421
1809
|
disabled: action.disabled,
|
1422
1810
|
onClick: handleClick(action),
|
1423
1811
|
justifyContent: "center",
|
1424
1812
|
variant: action.variant || "default",
|
1813
|
+
paddingTop: "7px",
|
1814
|
+
paddingBottom: "7px",
|
1425
1815
|
children: action.label
|
1426
1816
|
}
|
1427
1817
|
),
|
@@ -1429,7 +1819,7 @@ const DocumentActionButton = (action) => {
|
|
1429
1819
|
DocumentActionConfirmDialog,
|
1430
1820
|
{
|
1431
1821
|
...action.dialog,
|
1432
|
-
variant: action.variant,
|
1822
|
+
variant: action.dialog?.variant ?? action.variant,
|
1433
1823
|
isOpen: dialogId === action.id,
|
1434
1824
|
onClose: handleClose
|
1435
1825
|
}
|
@@ -1486,9 +1876,9 @@ const DocumentActionsMenu = ({
|
|
1486
1876
|
disabled: isDisabled,
|
1487
1877
|
size: "S",
|
1488
1878
|
endIcon: null,
|
1489
|
-
paddingTop: "
|
1490
|
-
paddingLeft: "
|
1491
|
-
paddingRight: "
|
1879
|
+
paddingTop: "4px",
|
1880
|
+
paddingLeft: "7px",
|
1881
|
+
paddingRight: "7px",
|
1492
1882
|
variant,
|
1493
1883
|
children: [
|
1494
1884
|
/* @__PURE__ */ jsx(More, { "aria-hidden": true, focusable: false }),
|
@@ -1499,7 +1889,7 @@ const DocumentActionsMenu = ({
|
|
1499
1889
|
]
|
1500
1890
|
}
|
1501
1891
|
),
|
1502
|
-
/* @__PURE__ */ jsxs(Menu.Content, {
|
1892
|
+
/* @__PURE__ */ jsxs(Menu.Content, { maxHeight: void 0, popoverPlacement: "bottom-end", children: [
|
1503
1893
|
actions2.map((action) => {
|
1504
1894
|
return /* @__PURE__ */ jsx(
|
1505
1895
|
Menu.Item,
|
@@ -1508,10 +1898,25 @@ const DocumentActionsMenu = ({
|
|
1508
1898
|
onSelect: handleClick(action),
|
1509
1899
|
display: "block",
|
1510
1900
|
children: /* @__PURE__ */ jsxs(Flex, { justifyContent: "space-between", gap: 4, children: [
|
1511
|
-
/* @__PURE__ */ jsxs(
|
1512
|
-
|
1513
|
-
|
1514
|
-
|
1901
|
+
/* @__PURE__ */ jsxs(
|
1902
|
+
Flex,
|
1903
|
+
{
|
1904
|
+
color: !action.disabled ? convertActionVariantToColor(action.variant) : "inherit",
|
1905
|
+
gap: 2,
|
1906
|
+
tag: "span",
|
1907
|
+
children: [
|
1908
|
+
/* @__PURE__ */ jsx(
|
1909
|
+
Flex,
|
1910
|
+
{
|
1911
|
+
tag: "span",
|
1912
|
+
color: !action.disabled ? convertActionVariantToIconColor(action.variant) : "inherit",
|
1913
|
+
children: action.icon
|
1914
|
+
}
|
1915
|
+
),
|
1916
|
+
action.label
|
1917
|
+
]
|
1918
|
+
}
|
1919
|
+
),
|
1515
1920
|
action.id.startsWith("HistoryAction") && /* @__PURE__ */ jsx(
|
1516
1921
|
Flex,
|
1517
1922
|
{
|
@@ -1570,6 +1975,18 @@ const convertActionVariantToColor = (variant = "secondary") => {
|
|
1570
1975
|
return "primary600";
|
1571
1976
|
}
|
1572
1977
|
};
|
1978
|
+
const convertActionVariantToIconColor = (variant = "secondary") => {
|
1979
|
+
switch (variant) {
|
1980
|
+
case "danger":
|
1981
|
+
return "danger600";
|
1982
|
+
case "secondary":
|
1983
|
+
return "neutral500";
|
1984
|
+
case "success":
|
1985
|
+
return "success600";
|
1986
|
+
default:
|
1987
|
+
return "primary600";
|
1988
|
+
}
|
1989
|
+
};
|
1573
1990
|
const DocumentActionConfirmDialog = ({
|
1574
1991
|
onClose,
|
1575
1992
|
onCancel,
|
@@ -1596,11 +2013,11 @@ const DocumentActionConfirmDialog = ({
|
|
1596
2013
|
/* @__PURE__ */ jsx(Dialog.Header, { children: title }),
|
1597
2014
|
/* @__PURE__ */ jsx(Dialog.Body, { children: content }),
|
1598
2015
|
/* @__PURE__ */ jsxs(Dialog.Footer, { children: [
|
1599
|
-
/* @__PURE__ */ jsx(Dialog.Cancel, { children: /* @__PURE__ */ jsx(Button, { variant: "tertiary", children: formatMessage({
|
2016
|
+
/* @__PURE__ */ jsx(Dialog.Cancel, { children: /* @__PURE__ */ jsx(Button, { variant: "tertiary", fullWidth: true, children: formatMessage({
|
1600
2017
|
id: "app.components.Button.cancel",
|
1601
2018
|
defaultMessage: "Cancel"
|
1602
2019
|
}) }) }),
|
1603
|
-
/* @__PURE__ */ jsx(Button, { onClick: handleConfirm, variant, children: formatMessage({
|
2020
|
+
/* @__PURE__ */ jsx(Button, { onClick: handleConfirm, variant, fullWidth: true, children: formatMessage({
|
1604
2021
|
id: "app.components.Button.confirm",
|
1605
2022
|
defaultMessage: "Confirm"
|
1606
2023
|
}) })
|
@@ -1639,13 +2056,17 @@ const PublishAction$1 = ({
|
|
1639
2056
|
const navigate = useNavigate();
|
1640
2057
|
const { toggleNotification } = useNotification();
|
1641
2058
|
const { _unstableFormatValidationErrors: formatValidationErrors } = useAPIErrorHandler();
|
2059
|
+
const isListView = useMatch(LIST_PATH) !== null;
|
1642
2060
|
const isCloning = useMatch(CLONE_PATH) !== null;
|
1643
2061
|
const { formatMessage } = useIntl();
|
1644
|
-
const { canPublish
|
1645
|
-
"PublishAction",
|
1646
|
-
({ canPublish: canPublish2, canCreate: canCreate2, canUpdate: canUpdate2 }) => ({ canPublish: canPublish2, canCreate: canCreate2, canUpdate: canUpdate2 })
|
1647
|
-
);
|
2062
|
+
const canPublish = useDocumentRBAC("PublishAction", ({ canPublish: canPublish2 }) => canPublish2);
|
1648
2063
|
const { publish } = useDocumentActions();
|
2064
|
+
const [
|
2065
|
+
countDraftRelations,
|
2066
|
+
{ isLoading: isLoadingDraftRelations, isError: isErrorDraftRelations }
|
2067
|
+
] = useLazyGetDraftRelationCountQuery();
|
2068
|
+
const [localCountOfDraftRelations, setLocalCountOfDraftRelations] = React.useState(0);
|
2069
|
+
const [serverCountOfDraftRelations, setServerCountOfDraftRelations] = React.useState(0);
|
1649
2070
|
const [{ query, rawQuery }] = useQueryParams();
|
1650
2071
|
const params = React.useMemo(() => buildValidParams(query), [query]);
|
1651
2072
|
const modified = useForm("PublishAction", ({ modified: modified2 }) => modified2);
|
@@ -1654,10 +2075,103 @@ const PublishAction$1 = ({
|
|
1654
2075
|
const validate = useForm("PublishAction", (state) => state.validate);
|
1655
2076
|
const setErrors = useForm("PublishAction", (state) => state.setErrors);
|
1656
2077
|
const formValues = useForm("PublishAction", ({ values }) => values);
|
1657
|
-
|
1658
|
-
|
2078
|
+
React.useEffect(() => {
|
2079
|
+
if (isErrorDraftRelations) {
|
2080
|
+
toggleNotification({
|
2081
|
+
type: "danger",
|
2082
|
+
message: formatMessage({
|
2083
|
+
id: getTranslation("error.records.fetch-draft-relatons"),
|
2084
|
+
defaultMessage: "An error occurred while fetching draft relations on this document."
|
2085
|
+
})
|
2086
|
+
});
|
2087
|
+
}
|
2088
|
+
}, [isErrorDraftRelations, toggleNotification, formatMessage]);
|
2089
|
+
React.useEffect(() => {
|
2090
|
+
const localDraftRelations = /* @__PURE__ */ new Set();
|
2091
|
+
const extractDraftRelations = (data) => {
|
2092
|
+
const relations = data.connect || [];
|
2093
|
+
relations.forEach((relation) => {
|
2094
|
+
if (relation.status === "draft") {
|
2095
|
+
localDraftRelations.add(relation.id);
|
2096
|
+
}
|
2097
|
+
});
|
2098
|
+
};
|
2099
|
+
const traverseAndExtract = (data) => {
|
2100
|
+
Object.entries(data).forEach(([key, value]) => {
|
2101
|
+
if (key === "connect" && Array.isArray(value)) {
|
2102
|
+
extractDraftRelations({ connect: value });
|
2103
|
+
} else if (typeof value === "object" && value !== null) {
|
2104
|
+
traverseAndExtract(value);
|
2105
|
+
}
|
2106
|
+
});
|
2107
|
+
};
|
2108
|
+
if (!documentId || modified) {
|
2109
|
+
traverseAndExtract(formValues);
|
2110
|
+
setLocalCountOfDraftRelations(localDraftRelations.size);
|
2111
|
+
}
|
2112
|
+
}, [documentId, modified, formValues, setLocalCountOfDraftRelations]);
|
2113
|
+
React.useEffect(() => {
|
2114
|
+
if (!document || !document.documentId || isListView) {
|
2115
|
+
return;
|
2116
|
+
}
|
2117
|
+
const fetchDraftRelationsCount = async () => {
|
2118
|
+
const { data, error } = await countDraftRelations({
|
2119
|
+
collectionType,
|
2120
|
+
model,
|
2121
|
+
documentId,
|
2122
|
+
params
|
2123
|
+
});
|
2124
|
+
if (error) {
|
2125
|
+
throw error;
|
2126
|
+
}
|
2127
|
+
if (data) {
|
2128
|
+
setServerCountOfDraftRelations(data.data);
|
2129
|
+
}
|
2130
|
+
};
|
2131
|
+
fetchDraftRelationsCount();
|
2132
|
+
}, [isListView, document, documentId, countDraftRelations, collectionType, model, params]);
|
2133
|
+
const isDocumentPublished = (document?.[PUBLISHED_AT_ATTRIBUTE_NAME] || meta?.availableStatus.some((doc) => doc[PUBLISHED_AT_ATTRIBUTE_NAME] !== null)) && document?.status !== "modified";
|
2134
|
+
if (!schema?.options?.draftAndPublish) {
|
1659
2135
|
return null;
|
1660
2136
|
}
|
2137
|
+
const performPublish = async () => {
|
2138
|
+
setSubmitting(true);
|
2139
|
+
try {
|
2140
|
+
const { errors } = await validate();
|
2141
|
+
if (errors) {
|
2142
|
+
toggleNotification({
|
2143
|
+
type: "danger",
|
2144
|
+
message: formatMessage({
|
2145
|
+
id: "content-manager.validation.error",
|
2146
|
+
defaultMessage: "There are validation errors in your document. Please fix them before saving."
|
2147
|
+
})
|
2148
|
+
});
|
2149
|
+
return;
|
2150
|
+
}
|
2151
|
+
const res = await publish(
|
2152
|
+
{
|
2153
|
+
collectionType,
|
2154
|
+
model,
|
2155
|
+
documentId,
|
2156
|
+
params
|
2157
|
+
},
|
2158
|
+
formValues
|
2159
|
+
);
|
2160
|
+
if ("data" in res && collectionType !== SINGLE_TYPES) {
|
2161
|
+
navigate({
|
2162
|
+
pathname: `../${collectionType}/${model}/${res.data.documentId}`,
|
2163
|
+
search: rawQuery
|
2164
|
+
});
|
2165
|
+
} else if ("error" in res && isBaseQueryError(res.error) && res.error.name === "ValidationError") {
|
2166
|
+
setErrors(formatValidationErrors(res.error));
|
2167
|
+
}
|
2168
|
+
} finally {
|
2169
|
+
setSubmitting(false);
|
2170
|
+
}
|
2171
|
+
};
|
2172
|
+
const totalDraftRelations = localCountOfDraftRelations + serverCountOfDraftRelations;
|
2173
|
+
const enableDraftRelationsCount = false;
|
2174
|
+
const hasDraftRelations = enableDraftRelationsCount;
|
1661
2175
|
return {
|
1662
2176
|
/**
|
1663
2177
|
* Disabled when:
|
@@ -1667,49 +2181,36 @@ const PublishAction$1 = ({
|
|
1667
2181
|
* - the document is already published & not modified
|
1668
2182
|
* - the document is being created & not modified
|
1669
2183
|
* - the user doesn't have the permission to publish
|
1670
|
-
* - the user doesn't have the permission to create a new document
|
1671
|
-
* - the user doesn't have the permission to update the document
|
1672
2184
|
*/
|
1673
|
-
disabled: isCloning || isSubmitting || activeTab === "published" || !modified && isDocumentPublished || !modified && !document?.documentId || !canPublish
|
2185
|
+
disabled: isCloning || isSubmitting || isLoadingDraftRelations || activeTab === "published" || !modified && isDocumentPublished || !modified && !document?.documentId || !canPublish,
|
1674
2186
|
label: formatMessage({
|
1675
2187
|
id: "app.utils.publish",
|
1676
2188
|
defaultMessage: "Publish"
|
1677
2189
|
}),
|
1678
2190
|
onClick: async () => {
|
1679
|
-
|
1680
|
-
|
1681
|
-
|
1682
|
-
|
1683
|
-
|
1684
|
-
|
1685
|
-
|
1686
|
-
|
1687
|
-
|
1688
|
-
|
1689
|
-
|
1690
|
-
|
1691
|
-
|
1692
|
-
|
1693
|
-
|
1694
|
-
|
1695
|
-
|
1696
|
-
documentId,
|
1697
|
-
params
|
1698
|
-
},
|
1699
|
-
formValues
|
1700
|
-
);
|
1701
|
-
if ("data" in res && collectionType !== SINGLE_TYPES) {
|
1702
|
-
navigate({
|
1703
|
-
pathname: `../${collectionType}/${model}/${res.data.documentId}`,
|
1704
|
-
search: rawQuery
|
1705
|
-
});
|
1706
|
-
} else if ("error" in res && isBaseQueryError(res.error) && res.error.name === "ValidationError") {
|
1707
|
-
setErrors(formatValidationErrors(res.error));
|
2191
|
+
await performPublish();
|
2192
|
+
},
|
2193
|
+
dialog: hasDraftRelations ? {
|
2194
|
+
type: "dialog",
|
2195
|
+
variant: "danger",
|
2196
|
+
footer: null,
|
2197
|
+
title: formatMessage({
|
2198
|
+
id: getTranslation(`popUpwarning.warning.bulk-has-draft-relations.title`),
|
2199
|
+
defaultMessage: "Confirmation"
|
2200
|
+
}),
|
2201
|
+
content: formatMessage(
|
2202
|
+
{
|
2203
|
+
id: getTranslation(`popUpwarning.warning.bulk-has-draft-relations.message`),
|
2204
|
+
defaultMessage: "This entry is related to {count, plural, one {# draft entry} other {# draft entries}}. Publishing it could leave broken links in your app."
|
2205
|
+
},
|
2206
|
+
{
|
2207
|
+
count: totalDraftRelations
|
1708
2208
|
}
|
1709
|
-
|
1710
|
-
|
2209
|
+
),
|
2210
|
+
onConfirm: async () => {
|
2211
|
+
await performPublish();
|
1711
2212
|
}
|
1712
|
-
}
|
2213
|
+
} : void 0
|
1713
2214
|
};
|
1714
2215
|
};
|
1715
2216
|
PublishAction$1.type = "publish";
|
@@ -1725,10 +2226,6 @@ const UpdateAction = ({
|
|
1725
2226
|
const cloneMatch = useMatch(CLONE_PATH);
|
1726
2227
|
const isCloning = cloneMatch !== null;
|
1727
2228
|
const { formatMessage } = useIntl();
|
1728
|
-
const { canCreate, canUpdate } = useDocumentRBAC("UpdateAction", ({ canCreate: canCreate2, canUpdate: canUpdate2 }) => ({
|
1729
|
-
canCreate: canCreate2,
|
1730
|
-
canUpdate: canUpdate2
|
1731
|
-
}));
|
1732
2229
|
const { create, update, clone } = useDocumentActions();
|
1733
2230
|
const [{ query, rawQuery }] = useQueryParams();
|
1734
2231
|
const params = React.useMemo(() => buildValidParams(query), [query]);
|
@@ -1745,10 +2242,8 @@ const UpdateAction = ({
|
|
1745
2242
|
* - the form is submitting
|
1746
2243
|
* - the document is not modified & we're not cloning (you can save a clone entity straight away)
|
1747
2244
|
* - the active tab is the published tab
|
1748
|
-
* - the user doesn't have the permission to create a new document
|
1749
|
-
* - the user doesn't have the permission to update the document
|
1750
2245
|
*/
|
1751
|
-
disabled: isSubmitting || !modified && !isCloning || activeTab === "published"
|
2246
|
+
disabled: isSubmitting || !modified && !isCloning || activeTab === "published",
|
1752
2247
|
label: formatMessage({
|
1753
2248
|
id: "content-manager.containers.Edit.save",
|
1754
2249
|
defaultMessage: "Save"
|
@@ -1756,16 +2251,18 @@ const UpdateAction = ({
|
|
1756
2251
|
onClick: async () => {
|
1757
2252
|
setSubmitting(true);
|
1758
2253
|
try {
|
1759
|
-
|
1760
|
-
|
1761
|
-
|
1762
|
-
|
1763
|
-
|
1764
|
-
|
1765
|
-
|
1766
|
-
|
1767
|
-
|
1768
|
-
|
2254
|
+
if (activeTab !== "draft") {
|
2255
|
+
const { errors } = await validate();
|
2256
|
+
if (errors) {
|
2257
|
+
toggleNotification({
|
2258
|
+
type: "danger",
|
2259
|
+
message: formatMessage({
|
2260
|
+
id: "content-manager.validation.error",
|
2261
|
+
defaultMessage: "There are validation errors in your document. Please fix them before saving."
|
2262
|
+
})
|
2263
|
+
});
|
2264
|
+
return;
|
2265
|
+
}
|
1769
2266
|
}
|
1770
2267
|
if (isCloning) {
|
1771
2268
|
const res = await clone(
|
@@ -1777,10 +2274,13 @@ const UpdateAction = ({
|
|
1777
2274
|
document
|
1778
2275
|
);
|
1779
2276
|
if ("data" in res) {
|
1780
|
-
navigate(
|
1781
|
-
|
1782
|
-
|
1783
|
-
|
2277
|
+
navigate(
|
2278
|
+
{
|
2279
|
+
pathname: `../${res.data.documentId}`,
|
2280
|
+
search: rawQuery
|
2281
|
+
},
|
2282
|
+
{ relative: "path" }
|
2283
|
+
);
|
1784
2284
|
} else if ("error" in res && isBaseQueryError(res.error) && res.error.name === "ValidationError") {
|
1785
2285
|
setErrors(formatValidationErrors(res.error));
|
1786
2286
|
}
|
@@ -1808,10 +2308,13 @@ const UpdateAction = ({
|
|
1808
2308
|
document
|
1809
2309
|
);
|
1810
2310
|
if ("data" in res && collectionType !== SINGLE_TYPES) {
|
1811
|
-
navigate(
|
1812
|
-
|
1813
|
-
|
1814
|
-
|
2311
|
+
navigate(
|
2312
|
+
{
|
2313
|
+
pathname: `../${res.data.documentId}`,
|
2314
|
+
search: rawQuery
|
2315
|
+
},
|
2316
|
+
{ replace: true, relative: "path" }
|
2317
|
+
);
|
1815
2318
|
} else if ("error" in res && isBaseQueryError(res.error) && res.error.name === "ValidationError") {
|
1816
2319
|
setErrors(formatValidationErrors(res.error));
|
1817
2320
|
}
|
@@ -1855,7 +2358,7 @@ const UnpublishAction$1 = ({
|
|
1855
2358
|
id: "app.utils.unpublish",
|
1856
2359
|
defaultMessage: "Unpublish"
|
1857
2360
|
}),
|
1858
|
-
icon: /* @__PURE__ */ jsx(
|
2361
|
+
icon: /* @__PURE__ */ jsx(Cross, {}),
|
1859
2362
|
onClick: async () => {
|
1860
2363
|
if (!documentId && collectionType !== SINGLE_TYPES || isDocumentModified) {
|
1861
2364
|
if (!documentId) {
|
@@ -1967,7 +2470,7 @@ const DiscardAction = ({
|
|
1967
2470
|
id: "content-manager.actions.discard.label",
|
1968
2471
|
defaultMessage: "Discard changes"
|
1969
2472
|
}),
|
1970
|
-
icon: /* @__PURE__ */ jsx(
|
2473
|
+
icon: /* @__PURE__ */ jsx(Cross, {}),
|
1971
2474
|
position: ["panel", "table-row"],
|
1972
2475
|
variant: "danger",
|
1973
2476
|
dialog: {
|
@@ -1995,11 +2498,6 @@ const DiscardAction = ({
|
|
1995
2498
|
};
|
1996
2499
|
};
|
1997
2500
|
DiscardAction.type = "discard";
|
1998
|
-
const StyledCrossCircle = styled(CrossCircle)`
|
1999
|
-
path {
|
2000
|
-
fill: currentColor;
|
2001
|
-
}
|
2002
|
-
`;
|
2003
2501
|
const DEFAULT_ACTIONS = [PublishAction$1, UpdateAction, UnpublishAction$1, DiscardAction];
|
2004
2502
|
const intervals = ["years", "months", "days", "hours", "minutes", "seconds"];
|
2005
2503
|
const RelativeTime = React.forwardRef(
|
@@ -2047,7 +2545,7 @@ const getDisplayName = ({
|
|
2047
2545
|
};
|
2048
2546
|
const capitalise = (str) => str.charAt(0).toUpperCase() + str.slice(1);
|
2049
2547
|
const DocumentStatus = ({ status = "draft", ...restProps }) => {
|
2050
|
-
const statusVariant = status === "draft" ? "
|
2548
|
+
const statusVariant = status === "draft" ? "secondary" : status === "published" ? "success" : "alternative";
|
2051
2549
|
return /* @__PURE__ */ jsx(Status, { ...restProps, showBullet: false, size: "S", variant: statusVariant, children: /* @__PURE__ */ jsx(Typography, { tag: "span", variant: "omega", fontWeight: "bold", children: capitalise(status) }) });
|
2052
2550
|
};
|
2053
2551
|
const Header = ({ isCreating, status, title: documentTitle = "Untitled" }) => {
|
@@ -2057,23 +2555,13 @@ const Header = ({ isCreating, status, title: documentTitle = "Untitled" }) => {
|
|
2057
2555
|
id: "content-manager.containers.edit.title.new",
|
2058
2556
|
defaultMessage: "Create an entry"
|
2059
2557
|
}) : documentTitle;
|
2060
|
-
return /* @__PURE__ */ jsxs(Flex, { direction: "column", alignItems: "flex-start", paddingTop:
|
2558
|
+
return /* @__PURE__ */ jsxs(Flex, { direction: "column", alignItems: "flex-start", paddingTop: 6, paddingBottom: 4, gap: 2, children: [
|
2061
2559
|
/* @__PURE__ */ jsx(BackButton, {}),
|
2062
|
-
/* @__PURE__ */ jsxs(
|
2063
|
-
|
2064
|
-
{
|
2065
|
-
|
2066
|
-
|
2067
|
-
paddingTop: 1,
|
2068
|
-
gap: "80px",
|
2069
|
-
alignItems: "flex-start",
|
2070
|
-
children: [
|
2071
|
-
/* @__PURE__ */ jsx(Typography, { variant: "alpha", tag: "h1", children: title }),
|
2072
|
-
/* @__PURE__ */ jsx(HeaderToolbar, {})
|
2073
|
-
]
|
2074
|
-
}
|
2075
|
-
),
|
2076
|
-
status ? /* @__PURE__ */ jsx(DocumentStatus, { status: isCloning ? "draft" : status }) : null
|
2560
|
+
/* @__PURE__ */ jsxs(Flex, { width: "100%", justifyContent: "space-between", gap: "80px", alignItems: "flex-start", children: [
|
2561
|
+
/* @__PURE__ */ jsx(Typography, { variant: "alpha", tag: "h1", children: title }),
|
2562
|
+
/* @__PURE__ */ jsx(HeaderToolbar, {})
|
2563
|
+
] }),
|
2564
|
+
status ? /* @__PURE__ */ jsx(Box, { marginTop: 1, children: /* @__PURE__ */ jsx(DocumentStatus, { status: isCloning ? "draft" : status }) }) : null
|
2077
2565
|
] });
|
2078
2566
|
};
|
2079
2567
|
const HeaderToolbar = () => {
|
@@ -2240,8 +2728,22 @@ const Information = ({ activeTab }) => {
|
|
2240
2728
|
);
|
2241
2729
|
};
|
2242
2730
|
const HeaderActions = ({ actions: actions2 }) => {
|
2243
|
-
|
2244
|
-
|
2731
|
+
const [dialogId, setDialogId] = React.useState(null);
|
2732
|
+
const handleClick = (action) => async (e) => {
|
2733
|
+
if (!("options" in action)) {
|
2734
|
+
const { onClick = () => false, dialog, id } = action;
|
2735
|
+
const muteDialog = await onClick(e);
|
2736
|
+
if (dialog && !muteDialog) {
|
2737
|
+
e.preventDefault();
|
2738
|
+
setDialogId(id);
|
2739
|
+
}
|
2740
|
+
}
|
2741
|
+
};
|
2742
|
+
const handleClose = () => {
|
2743
|
+
setDialogId(null);
|
2744
|
+
};
|
2745
|
+
return /* @__PURE__ */ jsx(Flex, { gap: 1, children: actions2.map((action) => {
|
2746
|
+
if (action.options) {
|
2245
2747
|
return /* @__PURE__ */ jsx(
|
2246
2748
|
SingleSelect,
|
2247
2749
|
{
|
@@ -2255,10 +2757,49 @@ const HeaderActions = ({ actions: actions2 }) => {
|
|
2255
2757
|
action.id
|
2256
2758
|
);
|
2257
2759
|
} else {
|
2258
|
-
|
2760
|
+
if (action.type === "icon") {
|
2761
|
+
return /* @__PURE__ */ jsxs(React.Fragment, { children: [
|
2762
|
+
/* @__PURE__ */ jsx(
|
2763
|
+
IconButton,
|
2764
|
+
{
|
2765
|
+
disabled: action.disabled,
|
2766
|
+
label: action.label,
|
2767
|
+
size: "S",
|
2768
|
+
onClick: handleClick(action),
|
2769
|
+
children: action.icon
|
2770
|
+
}
|
2771
|
+
),
|
2772
|
+
action.dialog ? /* @__PURE__ */ jsx(
|
2773
|
+
HeaderActionDialog,
|
2774
|
+
{
|
2775
|
+
...action.dialog,
|
2776
|
+
isOpen: dialogId === action.id,
|
2777
|
+
onClose: handleClose
|
2778
|
+
}
|
2779
|
+
) : null
|
2780
|
+
] }, action.id);
|
2781
|
+
}
|
2259
2782
|
}
|
2260
2783
|
}) });
|
2261
2784
|
};
|
2785
|
+
const HeaderActionDialog = ({
|
2786
|
+
onClose,
|
2787
|
+
onCancel,
|
2788
|
+
title,
|
2789
|
+
content: Content,
|
2790
|
+
isOpen
|
2791
|
+
}) => {
|
2792
|
+
const handleClose = async () => {
|
2793
|
+
if (onCancel) {
|
2794
|
+
await onCancel();
|
2795
|
+
}
|
2796
|
+
onClose();
|
2797
|
+
};
|
2798
|
+
return /* @__PURE__ */ jsx(Dialog.Root, { open: isOpen, onOpenChange: handleClose, children: /* @__PURE__ */ jsxs(Dialog.Content, { children: [
|
2799
|
+
/* @__PURE__ */ jsx(Dialog.Header, { children: title }),
|
2800
|
+
typeof Content === "function" ? /* @__PURE__ */ jsx(Content, { onClose: handleClose }) : Content
|
2801
|
+
] }) });
|
2802
|
+
};
|
2262
2803
|
const ConfigureTheViewAction = ({ collectionType, model }) => {
|
2263
2804
|
const navigate = useNavigate();
|
2264
2805
|
const { formatMessage } = useIntl();
|
@@ -2299,12 +2840,16 @@ const DeleteAction$1 = ({ documentId, model, collectionType, document }) => {
|
|
2299
2840
|
const { delete: deleteAction } = useDocumentActions();
|
2300
2841
|
const { toggleNotification } = useNotification();
|
2301
2842
|
const setSubmitting = useForm("DeleteAction", (state) => state.setSubmitting);
|
2843
|
+
const isLocalized = document?.locale != null;
|
2302
2844
|
return {
|
2303
2845
|
disabled: !canDelete || !document,
|
2304
|
-
label: formatMessage(
|
2305
|
-
|
2306
|
-
|
2307
|
-
|
2846
|
+
label: formatMessage(
|
2847
|
+
{
|
2848
|
+
id: "content-manager.actions.delete.label",
|
2849
|
+
defaultMessage: "Delete entry{isLocalized, select, true { (all locales)} other {}}"
|
2850
|
+
},
|
2851
|
+
{ isLocalized }
|
2852
|
+
),
|
2308
2853
|
icon: /* @__PURE__ */ jsx(Trash, {}),
|
2309
2854
|
dialog: {
|
2310
2855
|
type: "dialog",
|
@@ -2340,423 +2885,121 @@ const DeleteAction$1 = ({ documentId, model, collectionType, document }) => {
|
|
2340
2885
|
const res = await deleteAction({
|
2341
2886
|
documentId,
|
2342
2887
|
model,
|
2343
|
-
collectionType,
|
2344
|
-
params: {
|
2345
|
-
locale: "*"
|
2346
|
-
}
|
2347
|
-
});
|
2348
|
-
if (!("error" in res)) {
|
2349
|
-
navigate({ pathname: `../${collectionType}/${model}` }, { replace: true });
|
2350
|
-
}
|
2351
|
-
} finally {
|
2352
|
-
if (!listViewPathMatch) {
|
2353
|
-
setSubmitting(false);
|
2354
|
-
}
|
2355
|
-
}
|
2356
|
-
}
|
2357
|
-
},
|
2358
|
-
variant: "danger",
|
2359
|
-
position: ["header", "table-row"]
|
2360
|
-
};
|
2361
|
-
};
|
2362
|
-
DeleteAction$1.type = "delete";
|
2363
|
-
const DEFAULT_HEADER_ACTIONS = [EditTheModelAction, ConfigureTheViewAction, DeleteAction$1];
|
2364
|
-
const Panels = () => {
|
2365
|
-
const isCloning = useMatch(CLONE_PATH) !== null;
|
2366
|
-
const [
|
2367
|
-
{
|
2368
|
-
query: { status }
|
2369
|
-
}
|
2370
|
-
] = useQueryParams({
|
2371
|
-
status: "draft"
|
2372
|
-
});
|
2373
|
-
const { model, id, document, meta, collectionType } = useDoc();
|
2374
|
-
const plugins = useStrapiApp("Panels", (state) => state.plugins);
|
2375
|
-
const props = {
|
2376
|
-
activeTab: status,
|
2377
|
-
model,
|
2378
|
-
documentId: id,
|
2379
|
-
document: isCloning ? void 0 : document,
|
2380
|
-
meta: isCloning ? void 0 : meta,
|
2381
|
-
collectionType
|
2382
|
-
};
|
2383
|
-
return /* @__PURE__ */ jsx(Flex, { direction: "column", alignItems: "stretch", gap: 2, children: /* @__PURE__ */ jsx(
|
2384
|
-
DescriptionComponentRenderer,
|
2385
|
-
{
|
2386
|
-
props,
|
2387
|
-
descriptions: plugins["content-manager"].apis.getEditViewSidePanels(),
|
2388
|
-
children: (panels) => panels.map(({ content, id: id2, ...description }) => /* @__PURE__ */ jsx(Panel, { ...description, children: content }, id2))
|
2389
|
-
}
|
2390
|
-
) });
|
2391
|
-
};
|
2392
|
-
const ActionsPanel = () => {
|
2393
|
-
const { formatMessage } = useIntl();
|
2394
|
-
return {
|
2395
|
-
title: formatMessage({
|
2396
|
-
id: "content-manager.containers.edit.panels.default.title",
|
2397
|
-
defaultMessage: "Document"
|
2398
|
-
}),
|
2399
|
-
content: /* @__PURE__ */ jsx(ActionsPanelContent, {})
|
2400
|
-
};
|
2401
|
-
};
|
2402
|
-
ActionsPanel.type = "actions";
|
2403
|
-
const ActionsPanelContent = () => {
|
2404
|
-
const isCloning = useMatch(CLONE_PATH) !== null;
|
2405
|
-
const [
|
2406
|
-
{
|
2407
|
-
query: { status = "draft" }
|
2408
|
-
}
|
2409
|
-
] = useQueryParams();
|
2410
|
-
const { model, id, document, meta, collectionType } = useDoc();
|
2411
|
-
const plugins = useStrapiApp("ActionsPanel", (state) => state.plugins);
|
2412
|
-
const props = {
|
2413
|
-
activeTab: status,
|
2414
|
-
model,
|
2415
|
-
documentId: id,
|
2416
|
-
document: isCloning ? void 0 : document,
|
2417
|
-
meta: isCloning ? void 0 : meta,
|
2418
|
-
collectionType
|
2419
|
-
};
|
2420
|
-
return /* @__PURE__ */ jsxs(Flex, { direction: "column", gap: 2, width: "100%", children: [
|
2421
|
-
/* @__PURE__ */ jsx(
|
2422
|
-
DescriptionComponentRenderer,
|
2423
|
-
{
|
2424
|
-
props,
|
2425
|
-
descriptions: plugins["content-manager"].apis.getDocumentActions(),
|
2426
|
-
children: (actions2) => /* @__PURE__ */ jsx(DocumentActions, { actions: actions2 })
|
2427
|
-
}
|
2428
|
-
),
|
2429
|
-
/* @__PURE__ */ jsx(InjectionZone, { area: "editView.right-links", slug: model })
|
2430
|
-
] });
|
2431
|
-
};
|
2432
|
-
const Panel = React.forwardRef(({ children, title }, ref) => {
|
2433
|
-
return /* @__PURE__ */ jsxs(
|
2434
|
-
Flex,
|
2435
|
-
{
|
2436
|
-
ref,
|
2437
|
-
tag: "aside",
|
2438
|
-
"aria-labelledby": "additional-information",
|
2439
|
-
background: "neutral0",
|
2440
|
-
borderColor: "neutral150",
|
2441
|
-
hasRadius: true,
|
2442
|
-
paddingBottom: 4,
|
2443
|
-
paddingLeft: 4,
|
2444
|
-
paddingRight: 4,
|
2445
|
-
paddingTop: 4,
|
2446
|
-
shadow: "tableShadow",
|
2447
|
-
gap: 3,
|
2448
|
-
direction: "column",
|
2449
|
-
justifyContent: "stretch",
|
2450
|
-
alignItems: "flex-start",
|
2451
|
-
children: [
|
2452
|
-
/* @__PURE__ */ jsx(Typography, { tag: "h2", variant: "sigma", textTransform: "uppercase", children: title }),
|
2453
|
-
children
|
2454
|
-
]
|
2455
|
-
}
|
2456
|
-
);
|
2457
|
-
});
|
2458
|
-
const HOOKS = {
|
2459
|
-
/**
|
2460
|
-
* Hook that allows to mutate the displayed headers of the list view table
|
2461
|
-
* @constant
|
2462
|
-
* @type {string}
|
2463
|
-
*/
|
2464
|
-
INJECT_COLUMN_IN_TABLE: "Admin/CM/pages/ListView/inject-column-in-table",
|
2465
|
-
/**
|
2466
|
-
* Hook that allows to mutate the CM's collection types links pre-set filters
|
2467
|
-
* @constant
|
2468
|
-
* @type {string}
|
2469
|
-
*/
|
2470
|
-
MUTATE_COLLECTION_TYPES_LINKS: "Admin/CM/pages/App/mutate-collection-types-links",
|
2471
|
-
/**
|
2472
|
-
* Hook that allows to mutate the CM's edit view layout
|
2473
|
-
* @constant
|
2474
|
-
* @type {string}
|
2475
|
-
*/
|
2476
|
-
MUTATE_EDIT_VIEW_LAYOUT: "Admin/CM/pages/EditView/mutate-edit-view-layout",
|
2477
|
-
/**
|
2478
|
-
* Hook that allows to mutate the CM's single types links pre-set filters
|
2479
|
-
* @constant
|
2480
|
-
* @type {string}
|
2481
|
-
*/
|
2482
|
-
MUTATE_SINGLE_TYPES_LINKS: "Admin/CM/pages/App/mutate-single-types-links"
|
2483
|
-
};
|
2484
|
-
const contentTypesApi = contentManagerApi.injectEndpoints({
|
2485
|
-
endpoints: (builder) => ({
|
2486
|
-
getContentTypeConfiguration: builder.query({
|
2487
|
-
query: (uid) => ({
|
2488
|
-
url: `/content-manager/content-types/${uid}/configuration`,
|
2489
|
-
method: "GET"
|
2490
|
-
}),
|
2491
|
-
transformResponse: (response) => response.data,
|
2492
|
-
providesTags: (_result, _error, uid) => [
|
2493
|
-
{ type: "ContentTypesConfiguration", id: uid },
|
2494
|
-
{ type: "ContentTypeSettings", id: "LIST" }
|
2495
|
-
]
|
2496
|
-
}),
|
2497
|
-
getAllContentTypeSettings: builder.query({
|
2498
|
-
query: () => "/content-manager/content-types-settings",
|
2499
|
-
transformResponse: (response) => response.data,
|
2500
|
-
providesTags: [{ type: "ContentTypeSettings", id: "LIST" }]
|
2501
|
-
}),
|
2502
|
-
updateContentTypeConfiguration: builder.mutation({
|
2503
|
-
query: ({ uid, ...body }) => ({
|
2504
|
-
url: `/content-manager/content-types/${uid}/configuration`,
|
2505
|
-
method: "PUT",
|
2506
|
-
data: body
|
2507
|
-
}),
|
2508
|
-
transformResponse: (response) => response.data,
|
2509
|
-
invalidatesTags: (_result, _error, { uid }) => [
|
2510
|
-
{ type: "ContentTypesConfiguration", id: uid },
|
2511
|
-
{ type: "ContentTypeSettings", id: "LIST" },
|
2512
|
-
// Is this necessary?
|
2513
|
-
{ type: "InitialData" }
|
2514
|
-
]
|
2515
|
-
})
|
2516
|
-
})
|
2517
|
-
});
|
2518
|
-
const {
|
2519
|
-
useGetContentTypeConfigurationQuery,
|
2520
|
-
useGetAllContentTypeSettingsQuery,
|
2521
|
-
useUpdateContentTypeConfigurationMutation
|
2522
|
-
} = contentTypesApi;
|
2523
|
-
const checkIfAttributeIsDisplayable = (attribute) => {
|
2524
|
-
const { type } = attribute;
|
2525
|
-
if (type === "relation") {
|
2526
|
-
return !attribute.relation.toLowerCase().includes("morph");
|
2527
|
-
}
|
2528
|
-
return !["json", "dynamiczone", "richtext", "password", "blocks"].includes(type) && !!type;
|
2529
|
-
};
|
2530
|
-
const getMainField = (attribute, mainFieldName, { schemas, components }) => {
|
2531
|
-
if (!mainFieldName) {
|
2532
|
-
return void 0;
|
2533
|
-
}
|
2534
|
-
const mainFieldType = attribute.type === "component" ? components[attribute.component].attributes[mainFieldName].type : (
|
2535
|
-
// @ts-expect-error – `targetModel` does exist on the attribute for a relation.
|
2536
|
-
schemas.find((schema) => schema.uid === attribute.targetModel)?.attributes[mainFieldName].type
|
2537
|
-
);
|
2538
|
-
return {
|
2539
|
-
name: mainFieldName,
|
2540
|
-
type: mainFieldType ?? "string"
|
2541
|
-
};
|
2542
|
-
};
|
2543
|
-
const DEFAULT_SETTINGS = {
|
2544
|
-
bulkable: false,
|
2545
|
-
filterable: false,
|
2546
|
-
searchable: false,
|
2547
|
-
pagination: false,
|
2548
|
-
defaultSortBy: "",
|
2549
|
-
defaultSortOrder: "asc",
|
2550
|
-
mainField: "id",
|
2551
|
-
pageSize: 10
|
2552
|
-
};
|
2553
|
-
const useDocumentLayout = (model) => {
|
2554
|
-
const { schema, components } = useDocument({ model, collectionType: "" }, { skip: true });
|
2555
|
-
const [{ query }] = useQueryParams();
|
2556
|
-
const runHookWaterfall = useStrapiApp("useDocumentLayout", (state) => state.runHookWaterfall);
|
2557
|
-
const { toggleNotification } = useNotification();
|
2558
|
-
const { _unstableFormatAPIError: formatAPIError } = useAPIErrorHandler();
|
2559
|
-
const { isLoading: isLoadingSchemas, schemas } = useContentTypeSchema();
|
2560
|
-
const {
|
2561
|
-
data,
|
2562
|
-
isLoading: isLoadingConfigs,
|
2563
|
-
error,
|
2564
|
-
isFetching: isFetchingConfigs
|
2565
|
-
} = useGetContentTypeConfigurationQuery(model);
|
2566
|
-
const isLoading = isLoadingSchemas || isFetchingConfigs || isLoadingConfigs;
|
2567
|
-
React.useEffect(() => {
|
2568
|
-
if (error) {
|
2569
|
-
toggleNotification({
|
2570
|
-
type: "danger",
|
2571
|
-
message: formatAPIError(error)
|
2572
|
-
});
|
2573
|
-
}
|
2574
|
-
}, [error, formatAPIError, toggleNotification]);
|
2575
|
-
const editLayout = React.useMemo(
|
2576
|
-
() => data && !isLoading ? formatEditLayout(data, { schemas, schema, components }) : {
|
2577
|
-
layout: [],
|
2578
|
-
components: {},
|
2579
|
-
metadatas: {},
|
2580
|
-
options: {},
|
2581
|
-
settings: DEFAULT_SETTINGS
|
2582
|
-
},
|
2583
|
-
[data, isLoading, schemas, schema, components]
|
2584
|
-
);
|
2585
|
-
const listLayout = React.useMemo(() => {
|
2586
|
-
return data && !isLoading ? formatListLayout(data, { schemas, schema, components }) : {
|
2587
|
-
layout: [],
|
2588
|
-
metadatas: {},
|
2589
|
-
options: {},
|
2590
|
-
settings: DEFAULT_SETTINGS
|
2591
|
-
};
|
2592
|
-
}, [data, isLoading, schemas, schema, components]);
|
2593
|
-
const { layout: edit } = React.useMemo(
|
2594
|
-
() => runHookWaterfall(HOOKS.MUTATE_EDIT_VIEW_LAYOUT, {
|
2595
|
-
layout: editLayout,
|
2596
|
-
query
|
2597
|
-
}),
|
2598
|
-
[editLayout, query, runHookWaterfall]
|
2599
|
-
);
|
2600
|
-
return {
|
2601
|
-
error,
|
2602
|
-
isLoading,
|
2603
|
-
edit,
|
2604
|
-
list: listLayout
|
2605
|
-
};
|
2606
|
-
};
|
2607
|
-
const useDocLayout = () => {
|
2608
|
-
const { model } = useDoc();
|
2609
|
-
return useDocumentLayout(model);
|
2610
|
-
};
|
2611
|
-
const formatEditLayout = (data, {
|
2612
|
-
schemas,
|
2613
|
-
schema,
|
2614
|
-
components
|
2615
|
-
}) => {
|
2616
|
-
let currentPanelIndex = 0;
|
2617
|
-
const panelledEditAttributes = convertEditLayoutToFieldLayouts(
|
2618
|
-
data.contentType.layouts.edit,
|
2619
|
-
schema?.attributes,
|
2620
|
-
data.contentType.metadatas,
|
2621
|
-
{ configurations: data.components, schemas: components },
|
2622
|
-
schemas
|
2623
|
-
).reduce((panels, row) => {
|
2624
|
-
if (row.some((field) => field.type === "dynamiczone")) {
|
2625
|
-
panels.push([row]);
|
2626
|
-
currentPanelIndex += 2;
|
2627
|
-
} else {
|
2628
|
-
if (!panels[currentPanelIndex]) {
|
2629
|
-
panels.push([]);
|
2630
|
-
}
|
2631
|
-
panels[currentPanelIndex].push(row);
|
2632
|
-
}
|
2633
|
-
return panels;
|
2634
|
-
}, []);
|
2635
|
-
const componentEditAttributes = Object.entries(data.components).reduce(
|
2636
|
-
(acc, [uid, configuration]) => {
|
2637
|
-
acc[uid] = {
|
2638
|
-
layout: convertEditLayoutToFieldLayouts(
|
2639
|
-
configuration.layouts.edit,
|
2640
|
-
components[uid].attributes,
|
2641
|
-
configuration.metadatas
|
2642
|
-
),
|
2643
|
-
settings: {
|
2644
|
-
...configuration.settings,
|
2645
|
-
icon: components[uid].info.icon,
|
2646
|
-
displayName: components[uid].info.displayName
|
2888
|
+
collectionType,
|
2889
|
+
params: {
|
2890
|
+
locale: "*"
|
2891
|
+
}
|
2892
|
+
});
|
2893
|
+
if (!("error" in res)) {
|
2894
|
+
navigate({ pathname: `../${collectionType}/${model}` }, { replace: true });
|
2895
|
+
}
|
2896
|
+
} finally {
|
2897
|
+
if (!listViewPathMatch) {
|
2898
|
+
setSubmitting(false);
|
2899
|
+
}
|
2647
2900
|
}
|
2648
|
-
}
|
2649
|
-
return acc;
|
2650
|
-
},
|
2651
|
-
{}
|
2652
|
-
);
|
2653
|
-
const editMetadatas = Object.entries(data.contentType.metadatas).reduce(
|
2654
|
-
(acc, [attribute, metadata]) => {
|
2655
|
-
return {
|
2656
|
-
...acc,
|
2657
|
-
[attribute]: metadata.edit
|
2658
|
-
};
|
2659
|
-
},
|
2660
|
-
{}
|
2661
|
-
);
|
2662
|
-
return {
|
2663
|
-
layout: panelledEditAttributes,
|
2664
|
-
components: componentEditAttributes,
|
2665
|
-
metadatas: editMetadatas,
|
2666
|
-
settings: {
|
2667
|
-
...data.contentType.settings,
|
2668
|
-
displayName: schema?.info.displayName
|
2901
|
+
}
|
2669
2902
|
},
|
2670
|
-
|
2671
|
-
|
2672
|
-
...schema?.pluginOptions,
|
2673
|
-
...data.contentType.options
|
2674
|
-
}
|
2903
|
+
variant: "danger",
|
2904
|
+
position: ["header", "table-row"]
|
2675
2905
|
};
|
2676
2906
|
};
|
2677
|
-
|
2678
|
-
|
2679
|
-
|
2680
|
-
|
2681
|
-
|
2682
|
-
|
2683
|
-
}
|
2684
|
-
|
2685
|
-
|
2686
|
-
|
2687
|
-
|
2688
|
-
|
2689
|
-
|
2690
|
-
|
2691
|
-
|
2692
|
-
|
2693
|
-
|
2694
|
-
|
2695
|
-
|
2696
|
-
|
2697
|
-
|
2698
|
-
|
2699
|
-
|
2700
|
-
|
2701
|
-
|
2702
|
-
|
2703
|
-
}
|
2704
|
-
}
|
2705
|
-
);
|
2907
|
+
DeleteAction$1.type = "delete";
|
2908
|
+
const DEFAULT_HEADER_ACTIONS = [EditTheModelAction, ConfigureTheViewAction, DeleteAction$1];
|
2909
|
+
const Panels = () => {
|
2910
|
+
const isCloning = useMatch(CLONE_PATH) !== null;
|
2911
|
+
const [
|
2912
|
+
{
|
2913
|
+
query: { status }
|
2914
|
+
}
|
2915
|
+
] = useQueryParams({
|
2916
|
+
status: "draft"
|
2917
|
+
});
|
2918
|
+
const { model, id, document, meta, collectionType } = useDoc();
|
2919
|
+
const plugins = useStrapiApp("Panels", (state) => state.plugins);
|
2920
|
+
const props = {
|
2921
|
+
activeTab: status,
|
2922
|
+
model,
|
2923
|
+
documentId: id,
|
2924
|
+
document: isCloning ? void 0 : document,
|
2925
|
+
meta: isCloning ? void 0 : meta,
|
2926
|
+
collectionType
|
2927
|
+
};
|
2928
|
+
return /* @__PURE__ */ jsx(Flex, { direction: "column", alignItems: "stretch", gap: 2, children: /* @__PURE__ */ jsx(
|
2929
|
+
DescriptionComponentRenderer,
|
2930
|
+
{
|
2931
|
+
props,
|
2932
|
+
descriptions: plugins["content-manager"].apis.getEditViewSidePanels(),
|
2933
|
+
children: (panels) => panels.map(({ content, id: id2, ...description }) => /* @__PURE__ */ jsx(Panel, { ...description, children: content }, id2))
|
2934
|
+
}
|
2935
|
+
) });
|
2706
2936
|
};
|
2707
|
-
const
|
2708
|
-
|
2709
|
-
schema,
|
2710
|
-
components
|
2711
|
-
}) => {
|
2712
|
-
const listMetadatas = Object.entries(data.contentType.metadatas).reduce(
|
2713
|
-
(acc, [attribute, metadata]) => {
|
2714
|
-
return {
|
2715
|
-
...acc,
|
2716
|
-
[attribute]: metadata.list
|
2717
|
-
};
|
2718
|
-
},
|
2719
|
-
{}
|
2720
|
-
);
|
2721
|
-
const listAttributes = convertListLayoutToFieldLayouts(
|
2722
|
-
data.contentType.layouts.list,
|
2723
|
-
schema?.attributes,
|
2724
|
-
listMetadatas,
|
2725
|
-
{ configurations: data.components, schemas: components },
|
2726
|
-
schemas
|
2727
|
-
);
|
2937
|
+
const ActionsPanel = () => {
|
2938
|
+
const { formatMessage } = useIntl();
|
2728
2939
|
return {
|
2729
|
-
|
2730
|
-
|
2731
|
-
|
2732
|
-
|
2733
|
-
|
2734
|
-
...schema?.pluginOptions,
|
2735
|
-
...data.contentType.options
|
2736
|
-
}
|
2940
|
+
title: formatMessage({
|
2941
|
+
id: "content-manager.containers.edit.panels.default.title",
|
2942
|
+
defaultMessage: "Entry"
|
2943
|
+
}),
|
2944
|
+
content: /* @__PURE__ */ jsx(ActionsPanelContent, {})
|
2737
2945
|
};
|
2738
2946
|
};
|
2739
|
-
|
2740
|
-
|
2741
|
-
|
2742
|
-
|
2743
|
-
|
2947
|
+
ActionsPanel.type = "actions";
|
2948
|
+
const ActionsPanelContent = () => {
|
2949
|
+
const isCloning = useMatch(CLONE_PATH) !== null;
|
2950
|
+
const [
|
2951
|
+
{
|
2952
|
+
query: { status = "draft" }
|
2744
2953
|
}
|
2745
|
-
|
2746
|
-
|
2747
|
-
|
2748
|
-
|
2749
|
-
|
2750
|
-
|
2751
|
-
|
2752
|
-
|
2753
|
-
|
2754
|
-
|
2755
|
-
|
2756
|
-
|
2757
|
-
|
2758
|
-
|
2954
|
+
] = useQueryParams();
|
2955
|
+
const { model, id, document, meta, collectionType } = useDoc();
|
2956
|
+
const plugins = useStrapiApp("ActionsPanel", (state) => state.plugins);
|
2957
|
+
const props = {
|
2958
|
+
activeTab: status,
|
2959
|
+
model,
|
2960
|
+
documentId: id,
|
2961
|
+
document: isCloning ? void 0 : document,
|
2962
|
+
meta: isCloning ? void 0 : meta,
|
2963
|
+
collectionType
|
2964
|
+
};
|
2965
|
+
return /* @__PURE__ */ jsxs(Flex, { direction: "column", gap: 2, width: "100%", children: [
|
2966
|
+
/* @__PURE__ */ jsx(
|
2967
|
+
DescriptionComponentRenderer,
|
2968
|
+
{
|
2969
|
+
props,
|
2970
|
+
descriptions: plugins["content-manager"].apis.getDocumentActions(),
|
2971
|
+
children: (actions2) => /* @__PURE__ */ jsx(DocumentActions, { actions: actions2 })
|
2972
|
+
}
|
2973
|
+
),
|
2974
|
+
/* @__PURE__ */ jsx(InjectionZone, { area: "editView.right-links", slug: model })
|
2975
|
+
] });
|
2759
2976
|
};
|
2977
|
+
const Panel = React.forwardRef(({ children, title }, ref) => {
|
2978
|
+
return /* @__PURE__ */ jsxs(
|
2979
|
+
Flex,
|
2980
|
+
{
|
2981
|
+
ref,
|
2982
|
+
tag: "aside",
|
2983
|
+
"aria-labelledby": "additional-information",
|
2984
|
+
background: "neutral0",
|
2985
|
+
borderColor: "neutral150",
|
2986
|
+
hasRadius: true,
|
2987
|
+
paddingBottom: 4,
|
2988
|
+
paddingLeft: 4,
|
2989
|
+
paddingRight: 4,
|
2990
|
+
paddingTop: 4,
|
2991
|
+
shadow: "tableShadow",
|
2992
|
+
gap: 3,
|
2993
|
+
direction: "column",
|
2994
|
+
justifyContent: "stretch",
|
2995
|
+
alignItems: "flex-start",
|
2996
|
+
children: [
|
2997
|
+
/* @__PURE__ */ jsx(Typography, { tag: "h2", variant: "sigma", textTransform: "uppercase", children: title }),
|
2998
|
+
children
|
2999
|
+
]
|
3000
|
+
}
|
3001
|
+
);
|
3002
|
+
});
|
2760
3003
|
const ConfirmBulkActionDialog = ({
|
2761
3004
|
onToggleDialog,
|
2762
3005
|
isOpen = false,
|
@@ -2764,7 +3007,7 @@ const ConfirmBulkActionDialog = ({
|
|
2764
3007
|
endAction
|
2765
3008
|
}) => {
|
2766
3009
|
const { formatMessage } = useIntl();
|
2767
|
-
return /* @__PURE__ */ jsx(Dialog.Root, {
|
3010
|
+
return /* @__PURE__ */ jsx(Dialog.Root, { open: isOpen, children: /* @__PURE__ */ jsxs(Dialog.Content, { children: [
|
2768
3011
|
/* @__PURE__ */ jsx(Dialog.Header, { children: formatMessage({
|
2769
3012
|
id: "app.components.ConfirmDialog.title",
|
2770
3013
|
defaultMessage: "Confirmation"
|
@@ -2795,6 +3038,7 @@ const ConfirmDialogPublishAll = ({
|
|
2795
3038
|
const { _unstableFormatAPIError: formatAPIError } = useAPIErrorHandler(getTranslation);
|
2796
3039
|
const { model, schema } = useDoc();
|
2797
3040
|
const [{ query }] = useQueryParams();
|
3041
|
+
const enableDraftRelationsCount = false;
|
2798
3042
|
const {
|
2799
3043
|
data: countDraftRelations = 0,
|
2800
3044
|
isLoading,
|
@@ -2806,7 +3050,7 @@ const ConfirmDialogPublishAll = ({
|
|
2806
3050
|
locale: query?.plugins?.i18n?.locale
|
2807
3051
|
},
|
2808
3052
|
{
|
2809
|
-
skip:
|
3053
|
+
skip: !enableDraftRelationsCount
|
2810
3054
|
}
|
2811
3055
|
);
|
2812
3056
|
React.useEffect(() => {
|
@@ -2885,7 +3129,14 @@ const formatErrorMessages = (errors, parentKey, formatMessage) => {
|
|
2885
3129
|
)
|
2886
3130
|
);
|
2887
3131
|
} else {
|
2888
|
-
messages.push(
|
3132
|
+
messages.push(
|
3133
|
+
...formatErrorMessages(
|
3134
|
+
// @ts-expect-error TODO: check why value is not compatible with FormErrors
|
3135
|
+
value,
|
3136
|
+
currentKey,
|
3137
|
+
formatMessage
|
3138
|
+
)
|
3139
|
+
);
|
2889
3140
|
}
|
2890
3141
|
} else {
|
2891
3142
|
messages.push(
|
@@ -2984,7 +3235,7 @@ const SelectedEntriesTableContent = ({
|
|
2984
3235
|
status: row.status
|
2985
3236
|
}
|
2986
3237
|
) }),
|
2987
|
-
/* @__PURE__ */ jsx(Table.Cell, { children: /* @__PURE__ */ jsx(
|
3238
|
+
/* @__PURE__ */ jsx(Table.Cell, { children: /* @__PURE__ */ jsx(Flex, { children: /* @__PURE__ */ jsx(
|
2988
3239
|
IconButton,
|
2989
3240
|
{
|
2990
3241
|
tag: Link,
|
@@ -3007,9 +3258,10 @@ const SelectedEntriesTableContent = ({
|
|
3007
3258
|
),
|
3008
3259
|
target: "_blank",
|
3009
3260
|
marginLeft: "auto",
|
3010
|
-
|
3261
|
+
variant: "ghost",
|
3262
|
+
children: /* @__PURE__ */ jsx(Pencil, { width: "1.6rem", height: "1.6rem" })
|
3011
3263
|
}
|
3012
|
-
) })
|
3264
|
+
) }) })
|
3013
3265
|
] }, row.id)) })
|
3014
3266
|
] });
|
3015
3267
|
};
|
@@ -3046,7 +3298,13 @@ const SelectedEntriesModalContent = ({
|
|
3046
3298
|
);
|
3047
3299
|
const { rows, validationErrors } = React.useMemo(() => {
|
3048
3300
|
if (data.length > 0 && schema) {
|
3049
|
-
const validate = createYupSchema(
|
3301
|
+
const validate = createYupSchema(
|
3302
|
+
schema.attributes,
|
3303
|
+
components,
|
3304
|
+
// Since this is the "Publish" action, the validation
|
3305
|
+
// schema must enforce the rules for published entities
|
3306
|
+
{ status: "published" }
|
3307
|
+
);
|
3050
3308
|
const validationErrors2 = {};
|
3051
3309
|
const rows2 = data.map((entry) => {
|
3052
3310
|
try {
|
@@ -3396,7 +3654,7 @@ const TableActions = ({ document }) => {
|
|
3396
3654
|
DescriptionComponentRenderer,
|
3397
3655
|
{
|
3398
3656
|
props,
|
3399
|
-
descriptions: plugins["content-manager"].apis.getDocumentActions(),
|
3657
|
+
descriptions: plugins["content-manager"].apis.getDocumentActions().filter((action) => action.name !== "PublishAction"),
|
3400
3658
|
children: (actions2) => {
|
3401
3659
|
const tableRowActions = actions2.filter((action) => {
|
3402
3660
|
const positions = Array.isArray(action.position) ? action.position : [action.position];
|
@@ -3507,7 +3765,7 @@ const CloneAction = ({ model, documentId }) => {
|
|
3507
3765
|
}),
|
3508
3766
|
content: /* @__PURE__ */ jsx(AutoCloneFailureModalBody, { prohibitedFields }),
|
3509
3767
|
footer: ({ onClose }) => {
|
3510
|
-
return /* @__PURE__ */ jsxs(
|
3768
|
+
return /* @__PURE__ */ jsxs(Modal.Footer, { children: [
|
3511
3769
|
/* @__PURE__ */ jsx(Button, { onClick: onClose, variant: "tertiary", children: formatMessage({
|
3512
3770
|
id: "cancel",
|
3513
3771
|
defaultMessage: "Cancel"
|
@@ -3548,8 +3806,7 @@ class ContentManagerPlugin {
|
|
3548
3806
|
documentActions = [
|
3549
3807
|
...DEFAULT_ACTIONS,
|
3550
3808
|
...DEFAULT_TABLE_ROW_ACTIONS,
|
3551
|
-
...DEFAULT_HEADER_ACTIONS
|
3552
|
-
HistoryAction
|
3809
|
+
...DEFAULT_HEADER_ACTIONS
|
3553
3810
|
];
|
3554
3811
|
editViewSidePanels = [ActionsPanel];
|
3555
3812
|
headerActions = [];
|
@@ -3638,6 +3895,52 @@ const getPrintableType = (value) => {
|
|
3638
3895
|
}
|
3639
3896
|
return nativeType;
|
3640
3897
|
};
|
3898
|
+
const HistoryAction = ({ model, document }) => {
|
3899
|
+
const { formatMessage } = useIntl();
|
3900
|
+
const [{ query }] = useQueryParams();
|
3901
|
+
const navigate = useNavigate();
|
3902
|
+
const pluginsQueryParams = stringify({ plugins: query.plugins }, { encode: false });
|
3903
|
+
if (!window.strapi.features.isEnabled("cms-content-history")) {
|
3904
|
+
return null;
|
3905
|
+
}
|
3906
|
+
return {
|
3907
|
+
icon: /* @__PURE__ */ jsx(ClockCounterClockwise, {}),
|
3908
|
+
label: formatMessage({
|
3909
|
+
id: "content-manager.history.document-action",
|
3910
|
+
defaultMessage: "Content History"
|
3911
|
+
}),
|
3912
|
+
onClick: () => navigate({ pathname: "history", search: pluginsQueryParams }),
|
3913
|
+
disabled: (
|
3914
|
+
/**
|
3915
|
+
* The user is creating a new document.
|
3916
|
+
* It hasn't been saved yet, so there's no history to go to
|
3917
|
+
*/
|
3918
|
+
!document || /**
|
3919
|
+
* The document has been created but the current dimension has never been saved.
|
3920
|
+
* For example, the user is creating a new locale in an existing document,
|
3921
|
+
* so there's no history for the document in that locale
|
3922
|
+
*/
|
3923
|
+
!document.id || /**
|
3924
|
+
* History is only available for content types created by the user.
|
3925
|
+
* These have the `api::` prefix, as opposed to the ones created by Strapi or plugins,
|
3926
|
+
* which start with `admin::` or `plugin::`
|
3927
|
+
*/
|
3928
|
+
!model.startsWith("api::")
|
3929
|
+
),
|
3930
|
+
position: "header"
|
3931
|
+
};
|
3932
|
+
};
|
3933
|
+
HistoryAction.type = "history";
|
3934
|
+
const historyAdmin = {
|
3935
|
+
bootstrap(app) {
|
3936
|
+
const { addDocumentAction } = app.getPlugin("content-manager").apis;
|
3937
|
+
addDocumentAction((actions2) => {
|
3938
|
+
const indexOfDeleteAction = actions2.findIndex((action) => action.type === "delete");
|
3939
|
+
actions2.splice(indexOfDeleteAction, 0, HistoryAction);
|
3940
|
+
return actions2;
|
3941
|
+
});
|
3942
|
+
}
|
3943
|
+
};
|
3641
3944
|
const initialState = {
|
3642
3945
|
collectionTypeLinks: [],
|
3643
3946
|
components: [],
|
@@ -3688,15 +3991,29 @@ const index = {
|
|
3688
3991
|
defaultMessage: "Content Manager"
|
3689
3992
|
},
|
3690
3993
|
permissions: [],
|
3691
|
-
Component: () => import("./layout-DC503LnF.mjs").then((mod) => ({ default: mod.Layout })),
|
3692
3994
|
position: 1
|
3693
3995
|
});
|
3996
|
+
app.router.addRoute({
|
3997
|
+
path: "content-manager/*",
|
3998
|
+
lazy: async () => {
|
3999
|
+
const { Layout } = await import("./layout-B4UhJ8MJ.mjs");
|
4000
|
+
return {
|
4001
|
+
Component: Layout
|
4002
|
+
};
|
4003
|
+
},
|
4004
|
+
children: routes
|
4005
|
+
});
|
3694
4006
|
app.registerPlugin(cm.config);
|
3695
4007
|
},
|
4008
|
+
bootstrap(app) {
|
4009
|
+
if (typeof historyAdmin.bootstrap === "function") {
|
4010
|
+
historyAdmin.bootstrap(app);
|
4011
|
+
}
|
4012
|
+
},
|
3696
4013
|
async registerTrads({ locales }) {
|
3697
4014
|
const importedTrads = await Promise.all(
|
3698
4015
|
locales.map((locale) => {
|
3699
|
-
return __variableDynamicImportRuntimeHelper(/* @__PURE__ */ Object.assign({ "./translations/ar.json": () => import("./ar-CCEVvqGG.mjs"), "./translations/ca.json": () => import("./ca-5U32ON2v.mjs"), "./translations/cs.json": () => import("./cs-CM2aBUar.mjs"), "./translations/de.json": () => import("./de-C72KDNOl.mjs"), "./translations/en.json": () => import("./en-
|
4016
|
+
return __variableDynamicImportRuntimeHelper(/* @__PURE__ */ Object.assign({ "./translations/ar.json": () => import("./ar-CCEVvqGG.mjs"), "./translations/ca.json": () => import("./ca-5U32ON2v.mjs"), "./translations/cs.json": () => import("./cs-CM2aBUar.mjs"), "./translations/de.json": () => import("./de-C72KDNOl.mjs"), "./translations/en.json": () => import("./en-C8YBvRrK.mjs"), "./translations/es.json": () => import("./es-CeXiYflN.mjs"), "./translations/eu.json": () => import("./eu-CdALomew.mjs"), "./translations/fr.json": () => import("./fr-CD9VFbPM.mjs"), "./translations/gu.json": () => import("./gu-CNpaMDpH.mjs"), "./translations/hi.json": () => import("./hi-Dwvd04m3.mjs"), "./translations/hu.json": () => import("./hu-CeYvaaO0.mjs"), "./translations/id.json": () => import("./id-BtwA9WJT.mjs"), "./translations/it.json": () => import("./it-BrVPqaf1.mjs"), "./translations/ja.json": () => import("./ja-CtsUxOvk.mjs"), "./translations/ko.json": () => import("./ko-HVQRlfUI.mjs"), "./translations/ml.json": () => import("./ml-BihZwQit.mjs"), "./translations/ms.json": () => import("./ms-m_WjyWx7.mjs"), "./translations/nl.json": () => import("./nl-D4R9gHx5.mjs"), "./translations/pl.json": () => import("./pl-sbx9mSt_.mjs"), "./translations/pt-BR.json": () => import("./pt-BR-C71iDxnh.mjs"), "./translations/pt.json": () => import("./pt-BsaFvS8-.mjs"), "./translations/ru.json": () => import("./ru-BE6A4Exp.mjs"), "./translations/sa.json": () => import("./sa-Dag0k-Z8.mjs"), "./translations/sk.json": () => import("./sk-BFg-R8qJ.mjs"), "./translations/sv.json": () => import("./sv-CUnfWGsh.mjs"), "./translations/th.json": () => import("./th-BqbI8lIT.mjs"), "./translations/tr.json": () => import("./tr-CgeK3wJM.mjs"), "./translations/uk.json": () => import("./uk-CR-zDhAY.mjs"), "./translations/vi.json": () => import("./vi-DUXIk_fw.mjs"), "./translations/zh-Hans.json": () => import("./zh-Hans-BPQcRIyH.mjs"), "./translations/zh.json": () => import("./zh-BWZspA60.mjs") }), `./translations/${locale}.json`).then(({ default: data }) => {
|
3700
4017
|
return {
|
3701
4018
|
data: prefixPluginTranslations(data, PLUGIN_ID),
|
3702
4019
|
locale
|
@@ -3724,7 +4041,8 @@ export {
|
|
3724
4041
|
InjectionZone as I,
|
3725
4042
|
useDocument as J,
|
3726
4043
|
index as K,
|
3727
|
-
|
4044
|
+
useContentManagerContext as L,
|
4045
|
+
useDocumentActions as M,
|
3728
4046
|
Panels as P,
|
3729
4047
|
RelativeTime as R,
|
3730
4048
|
SINGLE_TYPES as S,
|
@@ -3742,11 +4060,11 @@ export {
|
|
3742
4060
|
PERMISSIONS as k,
|
3743
4061
|
DocumentRBAC as l,
|
3744
4062
|
DOCUMENT_META_FIELDS as m,
|
3745
|
-
|
3746
|
-
|
3747
|
-
|
3748
|
-
|
3749
|
-
|
4063
|
+
CLONE_PATH as n,
|
4064
|
+
useDocLayout as o,
|
4065
|
+
useGetContentTypeConfigurationQuery as p,
|
4066
|
+
CREATOR_FIELDS as q,
|
4067
|
+
getMainField as r,
|
3750
4068
|
setInitialData as s,
|
3751
4069
|
getDisplayName as t,
|
3752
4070
|
useContentTypeSchema as u,
|
@@ -3756,4 +4074,4 @@ export {
|
|
3756
4074
|
capitalise as y,
|
3757
4075
|
useUpdateContentTypeConfigurationMutation as z
|
3758
4076
|
};
|
3759
|
-
//# sourceMappingURL=index-
|
4077
|
+
//# sourceMappingURL=index-CPCHQ3X_.mjs.map
|