@strapi/content-manager 0.0.0-experimental.6d27139261823fc4b18da9f3c10b271d5010dbf0 → 0.0.0-experimental.7273fc63f5c2c8aa9e100655b1776f3e0621e61a
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/dist/_chunks/{ComponentConfigurationPage-DyDkPajU.js → ComponentConfigurationPage-D2_cVxXK.js} +4 -4
- package/dist/_chunks/{ComponentConfigurationPage-DyDkPajU.js.map → ComponentConfigurationPage-D2_cVxXK.js.map} +1 -1
- package/dist/_chunks/{ComponentConfigurationPage-9lRmRdIr.mjs → ComponentConfigurationPage-suSMW3KC.mjs} +4 -4
- package/dist/_chunks/{ComponentConfigurationPage-9lRmRdIr.mjs.map → ComponentConfigurationPage-suSMW3KC.mjs.map} +1 -1
- package/dist/_chunks/{EditConfigurationPage-Bk893vVY.mjs → EditConfigurationPage-CBscKbxt.mjs} +4 -4
- package/dist/_chunks/{EditConfigurationPage-Bk893vVY.mjs.map → EditConfigurationPage-CBscKbxt.mjs.map} +1 -1
- package/dist/_chunks/{EditConfigurationPage-DValmA0m.js → EditConfigurationPage-Z_VzHVH4.js} +4 -4
- package/dist/_chunks/{EditConfigurationPage-DValmA0m.js.map → EditConfigurationPage-Z_VzHVH4.js.map} +1 -1
- package/dist/_chunks/{EditViewPage-DiNFdFqP.mjs → EditViewPage-D5Zhyw_q.mjs} +15 -5
- package/dist/_chunks/EditViewPage-D5Zhyw_q.mjs.map +1 -0
- package/dist/_chunks/{EditViewPage-Dk7Eaft4.js → EditViewPage-DzDDKiF5.js} +15 -5
- package/dist/_chunks/EditViewPage-DzDDKiF5.js.map +1 -0
- package/dist/_chunks/{Field-DH2OaqUP.js → Field-CZHNLm_m.js} +77 -43
- package/dist/_chunks/Field-CZHNLm_m.js.map +1 -0
- package/dist/_chunks/{Field-Dv_HTFTa.mjs → Field-FNlyAbcI.mjs} +73 -39
- package/dist/_chunks/Field-FNlyAbcI.mjs.map +1 -0
- package/dist/_chunks/{Form-B_dUDizM.js → Form-DqPHDLuK.js} +16 -8
- package/dist/_chunks/Form-DqPHDLuK.js.map +1 -0
- package/dist/_chunks/{Form-Dy6P4HgH.mjs → Form-Sq_OVNv-.mjs} +16 -8
- package/dist/_chunks/Form-Sq_OVNv-.mjs.map +1 -0
- package/dist/_chunks/{History-BT4w83Oa.js → History-B-ZiOAEg.js} +37 -40
- package/dist/_chunks/History-B-ZiOAEg.js.map +1 -0
- package/dist/_chunks/{History-DrwsD1Vc.mjs → History-DlL3k_OW.mjs} +38 -41
- package/dist/_chunks/History-DlL3k_OW.mjs.map +1 -0
- package/dist/_chunks/{ListConfigurationPage-CuYrMcW3.js → ListConfigurationPage-Cc0hLAoS.js} +3 -3
- package/dist/_chunks/{ListConfigurationPage-CuYrMcW3.js.map → ListConfigurationPage-Cc0hLAoS.js.map} +1 -1
- package/dist/_chunks/{ListConfigurationPage-BxIP0jRy.mjs → ListConfigurationPage-xj3cSF5M.mjs} +3 -3
- package/dist/_chunks/{ListConfigurationPage-BxIP0jRy.mjs.map → ListConfigurationPage-xj3cSF5M.mjs.map} +1 -1
- package/dist/_chunks/{ListViewPage-5a1vw-OK.mjs → ListViewPage-C9STpcYd.mjs} +36 -13
- package/dist/_chunks/ListViewPage-C9STpcYd.mjs.map +1 -0
- package/dist/_chunks/{ListViewPage-BvpwNur7.js → ListViewPage-D48T84aT.js} +40 -17
- package/dist/_chunks/ListViewPage-D48T84aT.js.map +1 -0
- package/dist/_chunks/{NoContentTypePage-UqEiWKkM.js → NoContentTypePage-BnUmUnXQ.js} +2 -2
- package/dist/_chunks/{NoContentTypePage-UqEiWKkM.js.map → NoContentTypePage-BnUmUnXQ.js.map} +1 -1
- package/dist/_chunks/{NoContentTypePage-Bm6tRcd3.mjs → NoContentTypePage-DpV_ouD6.mjs} +2 -2
- package/dist/_chunks/{NoContentTypePage-Bm6tRcd3.mjs.map → NoContentTypePage-DpV_ouD6.mjs.map} +1 -1
- package/dist/_chunks/{NoPermissionsPage-BHPqn_tQ.mjs → NoPermissionsPage-BeDixsE9.mjs} +2 -2
- package/dist/_chunks/{NoPermissionsPage-BHPqn_tQ.mjs.map → NoPermissionsPage-BeDixsE9.mjs.map} +1 -1
- package/dist/_chunks/{NoPermissionsPage-C_vGRo8Q.js → NoPermissionsPage-z-CpOFfh.js} +2 -2
- package/dist/_chunks/{NoPermissionsPage-C_vGRo8Q.js.map → NoPermissionsPage-z-CpOFfh.js.map} +1 -1
- package/dist/_chunks/{Relations-C7fPyh5P.mjs → Relations-CTiimSAg.mjs} +72 -36
- package/dist/_chunks/Relations-CTiimSAg.mjs.map +1 -0
- package/dist/_chunks/{Relations-CznVF6LS.js → Relations-Cs5vJLAp.js} +71 -35
- package/dist/_chunks/Relations-Cs5vJLAp.js.map +1 -0
- package/dist/_chunks/{en-otD_UBJi.js → en-Bm0D0IWz.js} +12 -11
- package/dist/_chunks/{en-otD_UBJi.js.map → en-Bm0D0IWz.js.map} +1 -1
- package/dist/_chunks/{en-CbaIuYoB.mjs → en-DKV44jRb.mjs} +12 -11
- package/dist/_chunks/{en-CbaIuYoB.mjs.map → en-DKV44jRb.mjs.map} +1 -1
- package/dist/_chunks/{fr-B7kGGg3E.js → fr-C6t7ywEg.js} +12 -2
- package/dist/_chunks/{fr-B7kGGg3E.js.map → fr-C6t7ywEg.js.map} +1 -1
- package/dist/_chunks/{fr-CD9VFbPM.mjs → fr-DxrTDLl6.mjs} +12 -2
- package/dist/_chunks/{fr-CD9VFbPM.mjs.map → fr-DxrTDLl6.mjs.map} +1 -1
- package/dist/_chunks/{index-BJ6uTqLL.mjs → index-DUZzV11o.mjs} +1901 -1763
- package/dist/_chunks/index-DUZzV11o.mjs.map +1 -0
- package/dist/_chunks/{index-D9UmmBcM.js → index-dFQ_8EM3.js} +1898 -1759
- package/dist/_chunks/index-dFQ_8EM3.js.map +1 -0
- package/dist/_chunks/{layout-kfu5Wtix.js → layout-Bu-ET7w6.js} +5 -4
- package/dist/_chunks/{layout-kfu5Wtix.js.map → layout-Bu-ET7w6.js.map} +1 -1
- package/dist/_chunks/{layout-uomiIGbG.mjs → layout-C-eg7m9Y.mjs} +5 -4
- package/dist/_chunks/{layout-uomiIGbG.mjs.map → layout-C-eg7m9Y.mjs.map} +1 -1
- package/dist/_chunks/{objects-gigeqt7s.js → objects-BcXOv6_9.js} +2 -4
- package/dist/_chunks/{objects-gigeqt7s.js.map → objects-BcXOv6_9.js.map} +1 -1
- package/dist/_chunks/{objects-mKMAmfec.mjs → objects-D6yBsdmx.mjs} +2 -4
- package/dist/_chunks/{objects-mKMAmfec.mjs.map → objects-D6yBsdmx.mjs.map} +1 -1
- package/dist/_chunks/{relations-DiDufGSA.mjs → relations-Cj61Oyt2.mjs} +3 -7
- package/dist/_chunks/relations-Cj61Oyt2.mjs.map +1 -0
- package/dist/_chunks/{relations-DKENrxko.js → relations-isLeDaA0.js} +3 -7
- package/dist/_chunks/relations-isLeDaA0.js.map +1 -0
- 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/hooks/useDocument.d.ts +32 -1
- package/dist/admin/src/pages/EditView/components/Header.d.ts +11 -11
- package/dist/admin/src/preview/constants.d.ts +1 -0
- package/dist/admin/src/preview/index.d.ts +4 -0
- package/dist/admin/src/services/documents.d.ts +3 -1
- package/dist/server/index.js +378 -162
- package/dist/server/index.js.map +1 -1
- package/dist/server/index.mjs +378 -162
- package/dist/server/index.mjs.map +1 -1
- package/dist/server/src/bootstrap.d.ts.map +1 -1
- package/dist/server/src/controllers/collection-types.d.ts.map +1 -1
- package/dist/server/src/controllers/index.d.ts.map +1 -1
- package/dist/server/src/controllers/relations.d.ts.map +1 -1
- package/dist/server/src/controllers/utils/metadata.d.ts +15 -1
- package/dist/server/src/controllers/utils/metadata.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 -3
- package/dist/server/src/history/services/utils.d.ts.map +1 -1
- package/dist/server/src/index.d.ts +4 -4
- package/dist/server/src/preview/constants.d.ts +2 -0
- package/dist/server/src/preview/constants.d.ts.map +1 -0
- package/dist/server/src/preview/controllers/index.d.ts +2 -0
- package/dist/server/src/preview/controllers/index.d.ts.map +1 -0
- package/dist/server/src/preview/controllers/preview.d.ts +13 -0
- package/dist/server/src/preview/controllers/preview.d.ts.map +1 -0
- package/dist/server/src/preview/controllers/validation/preview.d.ts +6 -0
- package/dist/server/src/preview/controllers/validation/preview.d.ts.map +1 -0
- package/dist/server/src/preview/index.d.ts +4 -0
- package/dist/server/src/preview/index.d.ts.map +1 -0
- package/dist/server/src/preview/routes/index.d.ts +8 -0
- package/dist/server/src/preview/routes/index.d.ts.map +1 -0
- package/dist/server/src/preview/routes/preview.d.ts +4 -0
- package/dist/server/src/preview/routes/preview.d.ts.map +1 -0
- package/dist/server/src/preview/services/index.d.ts +15 -0
- package/dist/server/src/preview/services/index.d.ts.map +1 -0
- package/dist/server/src/preview/services/preview-config.d.ts +30 -0
- package/dist/server/src/preview/services/preview-config.d.ts.map +1 -0
- package/dist/server/src/preview/services/preview.d.ts +12 -0
- package/dist/server/src/preview/services/preview.d.ts.map +1 -0
- package/dist/server/src/preview/utils.d.ts +18 -0
- package/dist/server/src/preview/utils.d.ts.map +1 -0
- package/dist/server/src/routes/index.d.ts.map +1 -1
- package/dist/server/src/services/document-metadata.d.ts +8 -8
- package/dist/server/src/services/document-metadata.d.ts.map +1 -1
- package/dist/server/src/services/index.d.ts +4 -4
- package/dist/server/src/services/index.d.ts.map +1 -1
- package/dist/server/src/services/utils/configuration/index.d.ts +2 -2
- package/dist/server/src/services/utils/configuration/layouts.d.ts +2 -2
- package/dist/server/src/utils/index.d.ts +2 -0
- package/dist/server/src/utils/index.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/dist/shared/contracts/index.d.ts +1 -0
- package/dist/shared/contracts/index.d.ts.map +1 -1
- package/dist/shared/contracts/preview.d.ts +27 -0
- package/dist/shared/contracts/preview.d.ts.map +1 -0
- package/dist/shared/index.js +4 -0
- package/dist/shared/index.js.map +1 -1
- package/dist/shared/index.mjs +4 -0
- package/dist/shared/index.mjs.map +1 -1
- package/package.json +13 -13
- package/dist/_chunks/EditViewPage-DiNFdFqP.mjs.map +0 -1
- package/dist/_chunks/EditViewPage-Dk7Eaft4.js.map +0 -1
- package/dist/_chunks/Field-DH2OaqUP.js.map +0 -1
- package/dist/_chunks/Field-Dv_HTFTa.mjs.map +0 -1
- package/dist/_chunks/Form-B_dUDizM.js.map +0 -1
- package/dist/_chunks/Form-Dy6P4HgH.mjs.map +0 -1
- package/dist/_chunks/History-BT4w83Oa.js.map +0 -1
- package/dist/_chunks/History-DrwsD1Vc.mjs.map +0 -1
- package/dist/_chunks/ListViewPage-5a1vw-OK.mjs.map +0 -1
- package/dist/_chunks/ListViewPage-BvpwNur7.js.map +0 -1
- package/dist/_chunks/Relations-C7fPyh5P.mjs.map +0 -1
- package/dist/_chunks/Relations-CznVF6LS.js.map +0 -1
- package/dist/_chunks/index-BJ6uTqLL.mjs.map +0 -1
- package/dist/_chunks/index-D9UmmBcM.js.map +0 -1
- package/dist/_chunks/relations-DKENrxko.js.map +0 -1
- package/dist/_chunks/relations-DiDufGSA.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,16 +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, createContext, useAuth, useRBAC, Page, adminApi, translatedErrors, useNotification, useAPIErrorHandler, getYupValidationErrors,
|
3
|
+
import { useStrapiApp, createContext, useQueryParams, useAuth, useRBAC, Page, adminApi, translatedErrors, useNotification, useAPIErrorHandler, getYupValidationErrors, useForm, useTracking, useGuidedTour, BackButton, DescriptionComponentRenderer, useTable, Table } from "@strapi/admin/strapi-admin";
|
4
4
|
import * as React from "react";
|
5
5
|
import { lazy } from "react";
|
6
|
-
import { Button, Menu, VisuallyHidden, Flex, Typography, Dialog, Modal, Radio, Status, Box, SingleSelect, SingleSelectOption,
|
6
|
+
import { Button, Menu, VisuallyHidden, Flex, Typography, Dialog, Modal, Radio, Status, Box, SingleSelect, SingleSelectOption, IconButton, Loader, Tooltip, LinkButton } from "@strapi/design-system";
|
7
|
+
import mapValues from "lodash/fp/mapValues";
|
7
8
|
import { useIntl } from "react-intl";
|
8
9
|
import { useParams, useNavigate, Navigate, useMatch, useLocation, Link, NavLink } from "react-router-dom";
|
9
|
-
import { styled } from "styled-components";
|
10
10
|
import * as yup from "yup";
|
11
11
|
import { ValidationError } from "yup";
|
12
12
|
import pipe from "lodash/fp/pipe";
|
13
13
|
import { intervalToDuration, isPast } from "date-fns";
|
14
|
+
import { styled } from "styled-components";
|
14
15
|
import { stringify } from "qs";
|
15
16
|
import { createSlice, combineReducers } from "@reduxjs/toolkit";
|
16
17
|
const __variableDynamicImportRuntimeHelper = (glob, path) => {
|
@@ -100,6 +101,7 @@ const DocumentRBAC = ({ children, permissions }) => {
|
|
100
101
|
if (!slug) {
|
101
102
|
throw new Error("Cannot find the slug param in the URL");
|
102
103
|
}
|
104
|
+
const [{ rawQuery }] = useQueryParams();
|
103
105
|
const userPermissions = useAuth("DocumentRBAC", (state) => state.permissions);
|
104
106
|
const contentTypePermissions = React.useMemo(() => {
|
105
107
|
const contentTypePermissions2 = userPermissions.filter(
|
@@ -110,7 +112,14 @@ const DocumentRBAC = ({ children, permissions }) => {
|
|
110
112
|
return { ...acc, [action]: [permission] };
|
111
113
|
}, {});
|
112
114
|
}, [slug, userPermissions]);
|
113
|
-
const { isLoading, allowedActions } = useRBAC(
|
115
|
+
const { isLoading, allowedActions } = useRBAC(
|
116
|
+
contentTypePermissions,
|
117
|
+
permissions ?? void 0,
|
118
|
+
// TODO: useRBAC context should be typed and built differently
|
119
|
+
// We are passing raw query as context to the hook so that it can
|
120
|
+
// rely on the locale provided from DocumentRBAC for its permission calculations.
|
121
|
+
rawQuery
|
122
|
+
);
|
114
123
|
const canCreateFields = !isLoading && allowedActions.canCreate ? extractAndDedupeFields(contentTypePermissions.create) : [];
|
115
124
|
const canReadFields = !isLoading && allowedActions.canRead ? extractAndDedupeFields(contentTypePermissions.read) : [];
|
116
125
|
const canUpdateFields = !isLoading && allowedActions.canUpdate ? extractAndDedupeFields(contentTypePermissions.update) : [];
|
@@ -262,7 +271,7 @@ const documentApi = contentManagerApi.injectEndpoints({
|
|
262
271
|
*/
|
263
272
|
getAllDocuments: builder.query({
|
264
273
|
query: ({ model, params }) => ({
|
265
|
-
url: `/content-manager/collection-types/${model}`,
|
274
|
+
url: `/content-manager/collection-types/${model}${params ? `?${params}` : ""}`,
|
266
275
|
method: "GET",
|
267
276
|
config: {
|
268
277
|
params
|
@@ -461,6 +470,24 @@ const buildValidParams = (query) => {
|
|
461
470
|
const isBaseQueryError = (error) => {
|
462
471
|
return error.name !== void 0;
|
463
472
|
};
|
473
|
+
const arrayValidator = (attribute, options) => ({
|
474
|
+
message: translatedErrors.required,
|
475
|
+
test(value) {
|
476
|
+
if (options.status === "draft") {
|
477
|
+
return true;
|
478
|
+
}
|
479
|
+
if (!attribute.required) {
|
480
|
+
return true;
|
481
|
+
}
|
482
|
+
if (!value) {
|
483
|
+
return false;
|
484
|
+
}
|
485
|
+
if (Array.isArray(value) && value.length === 0) {
|
486
|
+
return false;
|
487
|
+
}
|
488
|
+
return true;
|
489
|
+
}
|
490
|
+
});
|
464
491
|
const createYupSchema = (attributes = {}, components = {}, options = { status: null }) => {
|
465
492
|
const createModelSchema = (attributes2) => yup.object().shape(
|
466
493
|
Object.entries(attributes2).reduce((acc, [name, attribute]) => {
|
@@ -468,6 +495,7 @@ const createYupSchema = (attributes = {}, components = {}, options = { status: n
|
|
468
495
|
return acc;
|
469
496
|
}
|
470
497
|
const validations = [
|
498
|
+
addNullableValidation,
|
471
499
|
addRequiredValidation,
|
472
500
|
addMinLengthValidation,
|
473
501
|
addMaxLengthValidation,
|
@@ -484,12 +512,12 @@ const createYupSchema = (attributes = {}, components = {}, options = { status: n
|
|
484
512
|
...acc,
|
485
513
|
[name]: transformSchema(
|
486
514
|
yup.array().of(createModelSchema(attributes3).nullable(false))
|
487
|
-
)
|
515
|
+
).test(arrayValidator(attribute, options))
|
488
516
|
};
|
489
517
|
} else {
|
490
518
|
return {
|
491
519
|
...acc,
|
492
|
-
[name]: transformSchema(createModelSchema(attributes3))
|
520
|
+
[name]: transformSchema(createModelSchema(attributes3).nullable())
|
493
521
|
};
|
494
522
|
}
|
495
523
|
}
|
@@ -511,7 +539,7 @@ const createYupSchema = (attributes = {}, components = {}, options = { status: n
|
|
511
539
|
}
|
512
540
|
)
|
513
541
|
)
|
514
|
-
)
|
542
|
+
).test(arrayValidator(attribute, options))
|
515
543
|
};
|
516
544
|
case "relation":
|
517
545
|
return {
|
@@ -523,7 +551,7 @@ const createYupSchema = (attributes = {}, components = {}, options = { status: n
|
|
523
551
|
} else if (Array.isArray(value)) {
|
524
552
|
return yup.array().of(
|
525
553
|
yup.object().shape({
|
526
|
-
id: yup.
|
554
|
+
id: yup.number().required()
|
527
555
|
})
|
528
556
|
);
|
529
557
|
} else if (typeof value === "object") {
|
@@ -609,17 +637,17 @@ const nullableSchema = (schema) => {
|
|
609
637
|
schema
|
610
638
|
);
|
611
639
|
};
|
640
|
+
const addNullableValidation = () => (schema) => {
|
641
|
+
return nullableSchema(schema);
|
642
|
+
};
|
612
643
|
const addRequiredValidation = (attribute, options) => (schema) => {
|
613
|
-
if (options.status === "draft") {
|
614
|
-
return
|
615
|
-
}
|
616
|
-
if ((attribute.type === "component" && attribute.repeatable || attribute.type === "dynamiczone") && attribute.required && "min" in schema) {
|
617
|
-
return schema.min(1, translatedErrors.required);
|
644
|
+
if (options.status === "draft" || !attribute.required) {
|
645
|
+
return schema;
|
618
646
|
}
|
619
|
-
if (attribute.required &&
|
647
|
+
if (attribute.required && "required" in schema) {
|
620
648
|
return schema.required(translatedErrors.required);
|
621
649
|
}
|
622
|
-
return
|
650
|
+
return schema;
|
623
651
|
};
|
624
652
|
const addMinLengthValidation = (attribute, options) => (schema) => {
|
625
653
|
if (options.status === "draft") {
|
@@ -647,31 +675,12 @@ const addMaxLengthValidation = (attribute) => (schema) => {
|
|
647
675
|
return schema;
|
648
676
|
};
|
649
677
|
const addMinValidation = (attribute, options) => (schema) => {
|
650
|
-
if ("
|
678
|
+
if (options.status === "draft") {
|
679
|
+
return schema;
|
680
|
+
}
|
681
|
+
if ("min" in attribute && "min" in schema) {
|
651
682
|
const min = toInteger(attribute.min);
|
652
|
-
if (
|
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
|
-
}
|
674
|
-
if ("min" in schema && min) {
|
683
|
+
if (min) {
|
675
684
|
return schema.min(min, {
|
676
685
|
...translatedErrors.min,
|
677
686
|
values: {
|
@@ -789,19 +798,115 @@ const extractContentTypeComponents = (attributes = {}, allComponents = {}) => {
|
|
789
798
|
}, {});
|
790
799
|
return componentsByKey;
|
791
800
|
};
|
792
|
-
const
|
801
|
+
const HOOKS = {
|
802
|
+
/**
|
803
|
+
* Hook that allows to mutate the displayed headers of the list view table
|
804
|
+
* @constant
|
805
|
+
* @type {string}
|
806
|
+
*/
|
807
|
+
INJECT_COLUMN_IN_TABLE: "Admin/CM/pages/ListView/inject-column-in-table",
|
808
|
+
/**
|
809
|
+
* Hook that allows to mutate the CM's collection types links pre-set filters
|
810
|
+
* @constant
|
811
|
+
* @type {string}
|
812
|
+
*/
|
813
|
+
MUTATE_COLLECTION_TYPES_LINKS: "Admin/CM/pages/App/mutate-collection-types-links",
|
814
|
+
/**
|
815
|
+
* Hook that allows to mutate the CM's edit view layout
|
816
|
+
* @constant
|
817
|
+
* @type {string}
|
818
|
+
*/
|
819
|
+
MUTATE_EDIT_VIEW_LAYOUT: "Admin/CM/pages/EditView/mutate-edit-view-layout",
|
820
|
+
/**
|
821
|
+
* Hook that allows to mutate the CM's single types links pre-set filters
|
822
|
+
* @constant
|
823
|
+
* @type {string}
|
824
|
+
*/
|
825
|
+
MUTATE_SINGLE_TYPES_LINKS: "Admin/CM/pages/App/mutate-single-types-links"
|
826
|
+
};
|
827
|
+
const contentTypesApi = contentManagerApi.injectEndpoints({
|
828
|
+
endpoints: (builder) => ({
|
829
|
+
getContentTypeConfiguration: builder.query({
|
830
|
+
query: (uid) => ({
|
831
|
+
url: `/content-manager/content-types/${uid}/configuration`,
|
832
|
+
method: "GET"
|
833
|
+
}),
|
834
|
+
transformResponse: (response) => response.data,
|
835
|
+
providesTags: (_result, _error, uid) => [
|
836
|
+
{ type: "ContentTypesConfiguration", id: uid },
|
837
|
+
{ type: "ContentTypeSettings", id: "LIST" }
|
838
|
+
]
|
839
|
+
}),
|
840
|
+
getAllContentTypeSettings: builder.query({
|
841
|
+
query: () => "/content-manager/content-types-settings",
|
842
|
+
transformResponse: (response) => response.data,
|
843
|
+
providesTags: [{ type: "ContentTypeSettings", id: "LIST" }]
|
844
|
+
}),
|
845
|
+
updateContentTypeConfiguration: builder.mutation({
|
846
|
+
query: ({ uid, ...body }) => ({
|
847
|
+
url: `/content-manager/content-types/${uid}/configuration`,
|
848
|
+
method: "PUT",
|
849
|
+
data: body
|
850
|
+
}),
|
851
|
+
transformResponse: (response) => response.data,
|
852
|
+
invalidatesTags: (_result, _error, { uid }) => [
|
853
|
+
{ type: "ContentTypesConfiguration", id: uid },
|
854
|
+
{ type: "ContentTypeSettings", id: "LIST" },
|
855
|
+
// Is this necessary?
|
856
|
+
{ type: "InitialData" }
|
857
|
+
]
|
858
|
+
})
|
859
|
+
})
|
860
|
+
});
|
861
|
+
const {
|
862
|
+
useGetContentTypeConfigurationQuery,
|
863
|
+
useGetAllContentTypeSettingsQuery,
|
864
|
+
useUpdateContentTypeConfigurationMutation
|
865
|
+
} = contentTypesApi;
|
866
|
+
const checkIfAttributeIsDisplayable = (attribute) => {
|
867
|
+
const { type } = attribute;
|
868
|
+
if (type === "relation") {
|
869
|
+
return !attribute.relation.toLowerCase().includes("morph");
|
870
|
+
}
|
871
|
+
return !["json", "dynamiczone", "richtext", "password", "blocks"].includes(type) && !!type;
|
872
|
+
};
|
873
|
+
const getMainField = (attribute, mainFieldName, { schemas, components }) => {
|
874
|
+
if (!mainFieldName) {
|
875
|
+
return void 0;
|
876
|
+
}
|
877
|
+
const mainFieldType = attribute.type === "component" ? components[attribute.component].attributes[mainFieldName].type : (
|
878
|
+
// @ts-expect-error – `targetModel` does exist on the attribute for a relation.
|
879
|
+
schemas.find((schema) => schema.uid === attribute.targetModel)?.attributes[mainFieldName].type
|
880
|
+
);
|
881
|
+
return {
|
882
|
+
name: mainFieldName,
|
883
|
+
type: mainFieldType ?? "string"
|
884
|
+
};
|
885
|
+
};
|
886
|
+
const DEFAULT_SETTINGS = {
|
887
|
+
bulkable: false,
|
888
|
+
filterable: false,
|
889
|
+
searchable: false,
|
890
|
+
pagination: false,
|
891
|
+
defaultSortBy: "",
|
892
|
+
defaultSortOrder: "asc",
|
893
|
+
mainField: "id",
|
894
|
+
pageSize: 10
|
895
|
+
};
|
896
|
+
const useDocumentLayout = (model) => {
|
897
|
+
const { schema, components } = useDocument({ model, collectionType: "" }, { skip: true });
|
898
|
+
const [{ query }] = useQueryParams();
|
899
|
+
const runHookWaterfall = useStrapiApp("useDocumentLayout", (state) => state.runHookWaterfall);
|
793
900
|
const { toggleNotification } = useNotification();
|
794
901
|
const { _unstableFormatAPIError: formatAPIError } = useAPIErrorHandler();
|
902
|
+
const { isLoading: isLoadingSchemas, schemas } = useContentTypeSchema();
|
795
903
|
const {
|
796
|
-
|
797
|
-
isLoading:
|
798
|
-
|
799
|
-
|
800
|
-
} =
|
801
|
-
|
802
|
-
skip: !args.documentId && args.collectionType !== SINGLE_TYPES || opts?.skip
|
803
|
-
});
|
804
|
-
const { components, schema, isLoading: isLoadingSchema } = useContentTypeSchema(args.model);
|
904
|
+
data,
|
905
|
+
isLoading: isLoadingConfigs,
|
906
|
+
error,
|
907
|
+
isFetching: isFetchingConfigs
|
908
|
+
} = useGetContentTypeConfigurationQuery(model);
|
909
|
+
const isLoading = isLoadingSchemas || isFetchingConfigs || isLoadingConfigs;
|
805
910
|
React.useEffect(() => {
|
806
911
|
if (error) {
|
807
912
|
toggleNotification({
|
@@ -809,396 +914,437 @@ const useDocument = (args, opts) => {
|
|
809
914
|
message: formatAPIError(error)
|
810
915
|
});
|
811
916
|
}
|
812
|
-
}, [
|
813
|
-
const
|
814
|
-
|
815
|
-
|
816
|
-
|
817
|
-
|
818
|
-
|
819
|
-
|
820
|
-
(document) => {
|
821
|
-
if (!validationSchema) {
|
822
|
-
throw new Error(
|
823
|
-
"There is no validation schema generated, this is likely due to the schema not being loaded yet."
|
824
|
-
);
|
825
|
-
}
|
826
|
-
try {
|
827
|
-
validationSchema.validateSync(document, { abortEarly: false, strict: true });
|
828
|
-
return null;
|
829
|
-
} catch (error2) {
|
830
|
-
if (error2 instanceof ValidationError) {
|
831
|
-
return getYupValidationErrors(error2);
|
832
|
-
}
|
833
|
-
throw error2;
|
834
|
-
}
|
917
|
+
}, [error, formatAPIError, toggleNotification]);
|
918
|
+
const editLayout = React.useMemo(
|
919
|
+
() => data && !isLoading ? formatEditLayout(data, { schemas, schema, components }) : {
|
920
|
+
layout: [],
|
921
|
+
components: {},
|
922
|
+
metadatas: {},
|
923
|
+
options: {},
|
924
|
+
settings: DEFAULT_SETTINGS
|
835
925
|
},
|
836
|
-
[
|
926
|
+
[data, isLoading, schemas, schema, components]
|
927
|
+
);
|
928
|
+
const listLayout = React.useMemo(() => {
|
929
|
+
return data && !isLoading ? formatListLayout(data, { schemas, schema, components }) : {
|
930
|
+
layout: [],
|
931
|
+
metadatas: {},
|
932
|
+
options: {},
|
933
|
+
settings: DEFAULT_SETTINGS
|
934
|
+
};
|
935
|
+
}, [data, isLoading, schemas, schema, components]);
|
936
|
+
const { layout: edit } = React.useMemo(
|
937
|
+
() => runHookWaterfall(HOOKS.MUTATE_EDIT_VIEW_LAYOUT, {
|
938
|
+
layout: editLayout,
|
939
|
+
query
|
940
|
+
}),
|
941
|
+
[editLayout, query, runHookWaterfall]
|
837
942
|
);
|
838
|
-
const isLoading = isLoadingDocument || isFetchingDocument || isLoadingSchema;
|
839
943
|
return {
|
840
|
-
|
841
|
-
document: data?.data,
|
842
|
-
meta: data?.meta,
|
944
|
+
error,
|
843
945
|
isLoading,
|
844
|
-
|
845
|
-
|
846
|
-
};
|
847
|
-
};
|
848
|
-
const useDoc = () => {
|
849
|
-
const { id, slug, collectionType, origin } = useParams();
|
850
|
-
const [{ query }] = useQueryParams();
|
851
|
-
const params = React.useMemo(() => buildValidParams(query), [query]);
|
852
|
-
if (!collectionType) {
|
853
|
-
throw new Error("Could not find collectionType in url params");
|
854
|
-
}
|
855
|
-
if (!slug) {
|
856
|
-
throw new Error("Could not find model in url params");
|
857
|
-
}
|
858
|
-
return {
|
859
|
-
collectionType,
|
860
|
-
model: slug,
|
861
|
-
id: origin || id === "create" ? void 0 : id,
|
862
|
-
...useDocument(
|
863
|
-
{ documentId: origin || id, model: slug, collectionType, params },
|
864
|
-
{
|
865
|
-
skip: id === "create" || !origin && !id && collectionType !== SINGLE_TYPES
|
866
|
-
}
|
867
|
-
)
|
946
|
+
edit,
|
947
|
+
list: listLayout
|
868
948
|
};
|
869
949
|
};
|
870
|
-
const
|
871
|
-
|
872
|
-
|
873
|
-
}
|
874
|
-
return Object.keys(trad).reduce((acc, current) => {
|
875
|
-
acc[`${pluginId}.${current}`] = trad[current];
|
876
|
-
return acc;
|
877
|
-
}, {});
|
878
|
-
};
|
879
|
-
const getTranslation = (id) => `content-manager.${id}`;
|
880
|
-
const DEFAULT_UNEXPECTED_ERROR_MSG = {
|
881
|
-
id: "notification.error",
|
882
|
-
defaultMessage: "An error occurred, please try again"
|
950
|
+
const useDocLayout = () => {
|
951
|
+
const { model } = useDoc();
|
952
|
+
return useDocumentLayout(model);
|
883
953
|
};
|
884
|
-
const
|
885
|
-
|
886
|
-
|
887
|
-
|
888
|
-
|
889
|
-
|
890
|
-
const
|
891
|
-
|
892
|
-
|
893
|
-
|
894
|
-
|
895
|
-
|
896
|
-
|
897
|
-
|
898
|
-
|
899
|
-
|
900
|
-
|
901
|
-
|
902
|
-
|
903
|
-
|
904
|
-
|
905
|
-
});
|
906
|
-
return { error: res.error };
|
907
|
-
}
|
908
|
-
toggleNotification({
|
909
|
-
type: "success",
|
910
|
-
message: formatMessage({
|
911
|
-
id: getTranslation("success.record.delete"),
|
912
|
-
defaultMessage: "Deleted document"
|
913
|
-
})
|
914
|
-
});
|
915
|
-
trackUsage("didDeleteEntry", trackerProperty);
|
916
|
-
return res.data;
|
917
|
-
} catch (err) {
|
918
|
-
toggleNotification({
|
919
|
-
type: "danger",
|
920
|
-
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
921
|
-
});
|
922
|
-
trackUsage("didNotDeleteEntry", { error: err, ...trackerProperty });
|
923
|
-
throw err;
|
954
|
+
const formatEditLayout = (data, {
|
955
|
+
schemas,
|
956
|
+
schema,
|
957
|
+
components
|
958
|
+
}) => {
|
959
|
+
let currentPanelIndex = 0;
|
960
|
+
const panelledEditAttributes = convertEditLayoutToFieldLayouts(
|
961
|
+
data.contentType.layouts.edit,
|
962
|
+
schema?.attributes,
|
963
|
+
data.contentType.metadatas,
|
964
|
+
{ configurations: data.components, schemas: components },
|
965
|
+
schemas
|
966
|
+
).reduce((panels, row) => {
|
967
|
+
if (row.some((field) => field.type === "dynamiczone")) {
|
968
|
+
panels.push([row]);
|
969
|
+
currentPanelIndex += 2;
|
970
|
+
} else {
|
971
|
+
if (!panels[currentPanelIndex]) {
|
972
|
+
panels.push([row]);
|
973
|
+
} else {
|
974
|
+
panels[currentPanelIndex].push(row);
|
924
975
|
}
|
925
|
-
}
|
926
|
-
|
927
|
-
);
|
928
|
-
const
|
929
|
-
|
930
|
-
|
931
|
-
|
932
|
-
|
933
|
-
|
934
|
-
|
935
|
-
|
936
|
-
|
937
|
-
|
938
|
-
|
939
|
-
|
940
|
-
|
941
|
-
message: formatAPIError(res.error)
|
942
|
-
});
|
943
|
-
return { error: res.error };
|
976
|
+
}
|
977
|
+
return panels;
|
978
|
+
}, []);
|
979
|
+
const componentEditAttributes = Object.entries(data.components).reduce(
|
980
|
+
(acc, [uid, configuration]) => {
|
981
|
+
acc[uid] = {
|
982
|
+
layout: convertEditLayoutToFieldLayouts(
|
983
|
+
configuration.layouts.edit,
|
984
|
+
components[uid].attributes,
|
985
|
+
configuration.metadatas,
|
986
|
+
{ configurations: data.components, schemas: components }
|
987
|
+
),
|
988
|
+
settings: {
|
989
|
+
...configuration.settings,
|
990
|
+
icon: components[uid].info.icon,
|
991
|
+
displayName: components[uid].info.displayName
|
944
992
|
}
|
945
|
-
|
946
|
-
|
947
|
-
title: formatMessage({
|
948
|
-
id: getTranslation("success.records.delete"),
|
949
|
-
defaultMessage: "Successfully deleted."
|
950
|
-
}),
|
951
|
-
message: ""
|
952
|
-
});
|
953
|
-
trackUsage("didBulkDeleteEntries");
|
954
|
-
return res.data;
|
955
|
-
} catch (err) {
|
956
|
-
toggleNotification({
|
957
|
-
type: "danger",
|
958
|
-
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
959
|
-
});
|
960
|
-
trackUsage("didNotBulkDeleteEntries");
|
961
|
-
throw err;
|
962
|
-
}
|
993
|
+
};
|
994
|
+
return acc;
|
963
995
|
},
|
964
|
-
|
996
|
+
{}
|
965
997
|
);
|
966
|
-
const
|
967
|
-
|
968
|
-
|
969
|
-
|
970
|
-
|
971
|
-
|
972
|
-
model,
|
973
|
-
documentId,
|
974
|
-
params
|
975
|
-
});
|
976
|
-
if ("error" in res) {
|
977
|
-
toggleNotification({
|
978
|
-
type: "danger",
|
979
|
-
message: formatAPIError(res.error)
|
980
|
-
});
|
981
|
-
return { error: res.error };
|
982
|
-
}
|
983
|
-
toggleNotification({
|
984
|
-
type: "success",
|
985
|
-
message: formatMessage({
|
986
|
-
id: "content-manager.success.record.discard",
|
987
|
-
defaultMessage: "Changes discarded"
|
988
|
-
})
|
989
|
-
});
|
990
|
-
return res.data;
|
991
|
-
} catch (err) {
|
992
|
-
toggleNotification({
|
993
|
-
type: "danger",
|
994
|
-
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
995
|
-
});
|
996
|
-
throw err;
|
997
|
-
}
|
998
|
+
const editMetadatas = Object.entries(data.contentType.metadatas).reduce(
|
999
|
+
(acc, [attribute, metadata]) => {
|
1000
|
+
return {
|
1001
|
+
...acc,
|
1002
|
+
[attribute]: metadata.edit
|
1003
|
+
};
|
998
1004
|
},
|
999
|
-
|
1005
|
+
{}
|
1000
1006
|
);
|
1001
|
-
|
1002
|
-
|
1003
|
-
|
1004
|
-
|
1005
|
-
|
1006
|
-
|
1007
|
-
|
1008
|
-
model,
|
1009
|
-
documentId,
|
1010
|
-
data,
|
1011
|
-
params
|
1012
|
-
});
|
1013
|
-
if ("error" in res) {
|
1014
|
-
toggleNotification({ type: "danger", message: formatAPIError(res.error) });
|
1015
|
-
return { error: res.error };
|
1016
|
-
}
|
1017
|
-
trackUsage("didPublishEntry");
|
1018
|
-
toggleNotification({
|
1019
|
-
type: "success",
|
1020
|
-
message: formatMessage({
|
1021
|
-
id: getTranslation("success.record.publish"),
|
1022
|
-
defaultMessage: "Published document"
|
1023
|
-
})
|
1024
|
-
});
|
1025
|
-
return res.data;
|
1026
|
-
} catch (err) {
|
1027
|
-
toggleNotification({
|
1028
|
-
type: "danger",
|
1029
|
-
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
1030
|
-
});
|
1031
|
-
throw err;
|
1032
|
-
}
|
1007
|
+
return {
|
1008
|
+
layout: panelledEditAttributes,
|
1009
|
+
components: componentEditAttributes,
|
1010
|
+
metadatas: editMetadatas,
|
1011
|
+
settings: {
|
1012
|
+
...data.contentType.settings,
|
1013
|
+
displayName: schema?.info.displayName
|
1033
1014
|
},
|
1034
|
-
|
1035
|
-
|
1036
|
-
|
1037
|
-
|
1038
|
-
|
1039
|
-
|
1040
|
-
|
1041
|
-
|
1042
|
-
|
1043
|
-
|
1044
|
-
|
1045
|
-
|
1046
|
-
|
1047
|
-
return { error: res.error };
|
1048
|
-
}
|
1049
|
-
toggleNotification({
|
1050
|
-
type: "success",
|
1051
|
-
message: formatMessage({
|
1052
|
-
id: getTranslation("success.record.publish"),
|
1053
|
-
defaultMessage: "Published document"
|
1054
|
-
})
|
1055
|
-
});
|
1056
|
-
return res.data;
|
1057
|
-
} catch (err) {
|
1058
|
-
toggleNotification({
|
1059
|
-
type: "danger",
|
1060
|
-
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
1061
|
-
});
|
1062
|
-
throw err;
|
1015
|
+
options: {
|
1016
|
+
...schema?.options,
|
1017
|
+
...schema?.pluginOptions,
|
1018
|
+
...data.contentType.options
|
1019
|
+
}
|
1020
|
+
};
|
1021
|
+
};
|
1022
|
+
const convertEditLayoutToFieldLayouts = (rows, attributes = {}, metadatas, components, schemas = []) => {
|
1023
|
+
return rows.map(
|
1024
|
+
(row) => row.map((field) => {
|
1025
|
+
const attribute = attributes[field.name];
|
1026
|
+
if (!attribute) {
|
1027
|
+
return null;
|
1063
1028
|
}
|
1064
|
-
|
1065
|
-
|
1066
|
-
|
1067
|
-
|
1068
|
-
|
1069
|
-
|
1070
|
-
|
1071
|
-
|
1029
|
+
const { edit: metadata } = metadatas[field.name];
|
1030
|
+
const settings = attribute.type === "component" && components ? components.configurations[attribute.component].settings : {};
|
1031
|
+
return {
|
1032
|
+
attribute,
|
1033
|
+
disabled: !metadata.editable,
|
1034
|
+
hint: metadata.description,
|
1035
|
+
label: metadata.label ?? "",
|
1036
|
+
name: field.name,
|
1037
|
+
// @ts-expect-error – mainField does exist on the metadata for a relation.
|
1038
|
+
mainField: getMainField(attribute, metadata.mainField || settings.mainField, {
|
1039
|
+
schemas,
|
1040
|
+
components: components?.schemas ?? {}
|
1041
|
+
}),
|
1042
|
+
placeholder: metadata.placeholder ?? "",
|
1043
|
+
required: attribute.required ?? false,
|
1044
|
+
size: field.size,
|
1045
|
+
unique: "unique" in attribute ? attribute.unique : false,
|
1046
|
+
visible: metadata.visible ?? true,
|
1047
|
+
type: attribute.type
|
1048
|
+
};
|
1049
|
+
}).filter((field) => field !== null)
|
1072
1050
|
);
|
1073
|
-
|
1074
|
-
|
1075
|
-
|
1076
|
-
|
1077
|
-
|
1078
|
-
|
1079
|
-
|
1080
|
-
|
1081
|
-
|
1082
|
-
|
1083
|
-
|
1084
|
-
|
1085
|
-
|
1086
|
-
|
1087
|
-
|
1088
|
-
|
1051
|
+
};
|
1052
|
+
const formatListLayout = (data, {
|
1053
|
+
schemas,
|
1054
|
+
schema,
|
1055
|
+
components
|
1056
|
+
}) => {
|
1057
|
+
const listMetadatas = Object.entries(data.contentType.metadatas).reduce(
|
1058
|
+
(acc, [attribute, metadata]) => {
|
1059
|
+
return {
|
1060
|
+
...acc,
|
1061
|
+
[attribute]: metadata.list
|
1062
|
+
};
|
1063
|
+
},
|
1064
|
+
{}
|
1065
|
+
);
|
1066
|
+
const listAttributes = convertListLayoutToFieldLayouts(
|
1067
|
+
data.contentType.layouts.list,
|
1068
|
+
schema?.attributes,
|
1069
|
+
listMetadatas,
|
1070
|
+
{ configurations: data.components, schemas: components },
|
1071
|
+
schemas
|
1072
|
+
);
|
1073
|
+
return {
|
1074
|
+
layout: listAttributes,
|
1075
|
+
settings: { ...data.contentType.settings, displayName: schema?.info.displayName },
|
1076
|
+
metadatas: listMetadatas,
|
1077
|
+
options: {
|
1078
|
+
...schema?.options,
|
1079
|
+
...schema?.pluginOptions,
|
1080
|
+
...data.contentType.options
|
1081
|
+
}
|
1082
|
+
};
|
1083
|
+
};
|
1084
|
+
const convertListLayoutToFieldLayouts = (columns, attributes = {}, metadatas, components, schemas = []) => {
|
1085
|
+
return columns.map((name) => {
|
1086
|
+
const attribute = attributes[name];
|
1087
|
+
if (!attribute) {
|
1088
|
+
return null;
|
1089
|
+
}
|
1090
|
+
const metadata = metadatas[name];
|
1091
|
+
const settings = attribute.type === "component" && components ? components.configurations[attribute.component].settings : {};
|
1092
|
+
return {
|
1093
|
+
attribute,
|
1094
|
+
label: metadata.label ?? "",
|
1095
|
+
mainField: getMainField(attribute, metadata.mainField || settings.mainField, {
|
1096
|
+
schemas,
|
1097
|
+
components: components?.schemas ?? {}
|
1098
|
+
}),
|
1099
|
+
name,
|
1100
|
+
searchable: metadata.searchable ?? true,
|
1101
|
+
sortable: metadata.sortable ?? true
|
1102
|
+
};
|
1103
|
+
}).filter((field) => field !== null);
|
1104
|
+
};
|
1105
|
+
const useDocument = (args, opts) => {
|
1106
|
+
const { toggleNotification } = useNotification();
|
1107
|
+
const { _unstableFormatAPIError: formatAPIError } = useAPIErrorHandler();
|
1108
|
+
const {
|
1109
|
+
currentData: data,
|
1110
|
+
isLoading: isLoadingDocument,
|
1111
|
+
isFetching: isFetchingDocument,
|
1112
|
+
error
|
1113
|
+
} = useGetDocumentQuery(args, {
|
1114
|
+
...opts,
|
1115
|
+
skip: !args.documentId && args.collectionType !== SINGLE_TYPES || opts?.skip
|
1116
|
+
});
|
1117
|
+
const {
|
1118
|
+
components,
|
1119
|
+
schema,
|
1120
|
+
schemas,
|
1121
|
+
isLoading: isLoadingSchema
|
1122
|
+
} = useContentTypeSchema(args.model);
|
1123
|
+
React.useEffect(() => {
|
1124
|
+
if (error) {
|
1125
|
+
toggleNotification({
|
1126
|
+
type: "danger",
|
1127
|
+
message: formatAPIError(error)
|
1128
|
+
});
|
1129
|
+
}
|
1130
|
+
}, [toggleNotification, error, formatAPIError, args.collectionType]);
|
1131
|
+
const validationSchema = React.useMemo(() => {
|
1132
|
+
if (!schema) {
|
1133
|
+
return null;
|
1134
|
+
}
|
1135
|
+
return createYupSchema(schema.attributes, components);
|
1136
|
+
}, [schema, components]);
|
1137
|
+
const validate = React.useCallback(
|
1138
|
+
(document) => {
|
1139
|
+
if (!validationSchema) {
|
1140
|
+
throw new Error(
|
1141
|
+
"There is no validation schema generated, this is likely due to the schema not being loaded yet."
|
1142
|
+
);
|
1143
|
+
}
|
1144
|
+
try {
|
1145
|
+
validationSchema.validateSync(document, { abortEarly: false, strict: true });
|
1146
|
+
return null;
|
1147
|
+
} catch (error2) {
|
1148
|
+
if (error2 instanceof ValidationError) {
|
1149
|
+
return getYupValidationErrors(error2);
|
1089
1150
|
}
|
1090
|
-
|
1091
|
-
toggleNotification({
|
1092
|
-
type: "success",
|
1093
|
-
message: formatMessage({
|
1094
|
-
id: getTranslation("success.record.save"),
|
1095
|
-
defaultMessage: "Saved document"
|
1096
|
-
})
|
1097
|
-
});
|
1098
|
-
return res.data;
|
1099
|
-
} catch (err) {
|
1100
|
-
trackUsage("didNotEditEntry", { error: err, ...trackerProperty });
|
1101
|
-
toggleNotification({
|
1102
|
-
type: "danger",
|
1103
|
-
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
1104
|
-
});
|
1105
|
-
throw err;
|
1151
|
+
throw error2;
|
1106
1152
|
}
|
1107
1153
|
},
|
1108
|
-
[
|
1154
|
+
[validationSchema]
|
1109
1155
|
);
|
1110
|
-
const
|
1111
|
-
const
|
1112
|
-
|
1156
|
+
const isLoading = isLoadingDocument || isFetchingDocument || isLoadingSchema;
|
1157
|
+
const hasError = !!error;
|
1158
|
+
return {
|
1159
|
+
components,
|
1160
|
+
document: data?.data,
|
1161
|
+
meta: data?.meta,
|
1162
|
+
isLoading,
|
1163
|
+
hasError,
|
1164
|
+
schema,
|
1165
|
+
schemas,
|
1166
|
+
validate
|
1167
|
+
};
|
1168
|
+
};
|
1169
|
+
const useDoc = () => {
|
1170
|
+
const { id, slug, collectionType, origin } = useParams();
|
1171
|
+
const [{ query }] = useQueryParams();
|
1172
|
+
const params = React.useMemo(() => buildValidParams(query), [query]);
|
1173
|
+
if (!collectionType) {
|
1174
|
+
throw new Error("Could not find collectionType in url params");
|
1175
|
+
}
|
1176
|
+
if (!slug) {
|
1177
|
+
throw new Error("Could not find model in url params");
|
1178
|
+
}
|
1179
|
+
const document = useDocument(
|
1180
|
+
{ documentId: origin || id, model: slug, collectionType, params },
|
1181
|
+
{
|
1182
|
+
skip: id === "create" || !origin && !id && collectionType !== SINGLE_TYPES
|
1183
|
+
}
|
1184
|
+
);
|
1185
|
+
const returnId = origin || id === "create" ? void 0 : id;
|
1186
|
+
return {
|
1187
|
+
collectionType,
|
1188
|
+
model: slug,
|
1189
|
+
id: returnId,
|
1190
|
+
...document
|
1191
|
+
};
|
1192
|
+
};
|
1193
|
+
const useContentManagerContext = () => {
|
1194
|
+
const {
|
1195
|
+
collectionType,
|
1196
|
+
model,
|
1197
|
+
id,
|
1198
|
+
components,
|
1199
|
+
isLoading: isLoadingDoc,
|
1200
|
+
schema,
|
1201
|
+
schemas
|
1202
|
+
} = useDoc();
|
1203
|
+
const layout = useDocumentLayout(model);
|
1204
|
+
const form = useForm("useContentManagerContext", (state) => state);
|
1205
|
+
const isSingleType = collectionType === SINGLE_TYPES;
|
1206
|
+
const slug = model;
|
1207
|
+
const isCreatingEntry = id === "create";
|
1208
|
+
useContentTypeSchema();
|
1209
|
+
const isLoading = isLoadingDoc || layout.isLoading;
|
1210
|
+
const error = layout.error;
|
1211
|
+
return {
|
1212
|
+
error,
|
1213
|
+
isLoading,
|
1214
|
+
// Base metadata
|
1215
|
+
model,
|
1216
|
+
collectionType,
|
1217
|
+
id,
|
1218
|
+
slug,
|
1219
|
+
isCreatingEntry,
|
1220
|
+
isSingleType,
|
1221
|
+
hasDraftAndPublish: schema?.options?.draftAndPublish ?? false,
|
1222
|
+
// All schema infos
|
1223
|
+
components,
|
1224
|
+
contentType: schema,
|
1225
|
+
contentTypes: schemas,
|
1226
|
+
// Form state
|
1227
|
+
form,
|
1228
|
+
// layout infos
|
1229
|
+
layout
|
1230
|
+
};
|
1231
|
+
};
|
1232
|
+
const prefixPluginTranslations = (trad, pluginId) => {
|
1233
|
+
return Object.keys(trad).reduce((acc, current) => {
|
1234
|
+
acc[`${pluginId}.${current}`] = trad[current];
|
1235
|
+
return acc;
|
1236
|
+
}, {});
|
1237
|
+
};
|
1238
|
+
const getTranslation = (id) => `content-manager.${id}`;
|
1239
|
+
const DEFAULT_UNEXPECTED_ERROR_MSG = {
|
1240
|
+
id: "notification.error",
|
1241
|
+
defaultMessage: "An error occurred, please try again"
|
1242
|
+
};
|
1243
|
+
const useDocumentActions = () => {
|
1244
|
+
const { toggleNotification } = useNotification();
|
1245
|
+
const { formatMessage } = useIntl();
|
1246
|
+
const { trackUsage } = useTracking();
|
1247
|
+
const { _unstableFormatAPIError: formatAPIError } = useAPIErrorHandler();
|
1248
|
+
const navigate = useNavigate();
|
1249
|
+
const setCurrentStep = useGuidedTour("useDocumentActions", (state) => state.setCurrentStep);
|
1250
|
+
const [deleteDocument] = useDeleteDocumentMutation();
|
1251
|
+
const _delete = React.useCallback(
|
1252
|
+
async ({ collectionType, model, documentId, params }, trackerProperty) => {
|
1113
1253
|
try {
|
1114
|
-
trackUsage("
|
1115
|
-
const res = await
|
1254
|
+
trackUsage("willDeleteEntry", trackerProperty);
|
1255
|
+
const res = await deleteDocument({
|
1116
1256
|
collectionType,
|
1117
1257
|
model,
|
1118
1258
|
documentId,
|
1119
|
-
params
|
1120
|
-
data: {
|
1121
|
-
discardDraft
|
1122
|
-
}
|
1259
|
+
params
|
1123
1260
|
});
|
1124
1261
|
if ("error" in res) {
|
1125
|
-
toggleNotification({
|
1262
|
+
toggleNotification({
|
1263
|
+
type: "danger",
|
1264
|
+
message: formatAPIError(res.error)
|
1265
|
+
});
|
1126
1266
|
return { error: res.error };
|
1127
1267
|
}
|
1128
|
-
trackUsage("didUnpublishEntry");
|
1129
1268
|
toggleNotification({
|
1130
1269
|
type: "success",
|
1131
1270
|
message: formatMessage({
|
1132
|
-
id: getTranslation("success.record.
|
1133
|
-
defaultMessage: "
|
1271
|
+
id: getTranslation("success.record.delete"),
|
1272
|
+
defaultMessage: "Deleted document"
|
1134
1273
|
})
|
1135
1274
|
});
|
1275
|
+
trackUsage("didDeleteEntry", trackerProperty);
|
1136
1276
|
return res.data;
|
1137
1277
|
} catch (err) {
|
1138
1278
|
toggleNotification({
|
1139
1279
|
type: "danger",
|
1140
1280
|
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
1141
1281
|
});
|
1282
|
+
trackUsage("didNotDeleteEntry", { error: err, ...trackerProperty });
|
1142
1283
|
throw err;
|
1143
1284
|
}
|
1144
1285
|
},
|
1145
|
-
[trackUsage,
|
1286
|
+
[trackUsage, deleteDocument, toggleNotification, formatMessage, formatAPIError]
|
1146
1287
|
);
|
1147
|
-
const [
|
1148
|
-
const
|
1288
|
+
const [deleteManyDocuments] = useDeleteManyDocumentsMutation();
|
1289
|
+
const deleteMany = React.useCallback(
|
1149
1290
|
async ({ model, documentIds, params }) => {
|
1150
1291
|
try {
|
1151
|
-
trackUsage("
|
1152
|
-
const res = await
|
1292
|
+
trackUsage("willBulkDeleteEntries");
|
1293
|
+
const res = await deleteManyDocuments({
|
1153
1294
|
model,
|
1154
1295
|
documentIds,
|
1155
1296
|
params
|
1156
1297
|
});
|
1157
1298
|
if ("error" in res) {
|
1158
|
-
toggleNotification({
|
1299
|
+
toggleNotification({
|
1300
|
+
type: "danger",
|
1301
|
+
message: formatAPIError(res.error)
|
1302
|
+
});
|
1159
1303
|
return { error: res.error };
|
1160
1304
|
}
|
1161
|
-
trackUsage("didBulkUnpublishEntries");
|
1162
1305
|
toggleNotification({
|
1163
1306
|
type: "success",
|
1164
1307
|
title: formatMessage({
|
1165
|
-
id: getTranslation("success.records.
|
1166
|
-
defaultMessage: "Successfully
|
1308
|
+
id: getTranslation("success.records.delete"),
|
1309
|
+
defaultMessage: "Successfully deleted."
|
1167
1310
|
}),
|
1168
1311
|
message: ""
|
1169
1312
|
});
|
1313
|
+
trackUsage("didBulkDeleteEntries");
|
1170
1314
|
return res.data;
|
1171
1315
|
} catch (err) {
|
1172
1316
|
toggleNotification({
|
1173
1317
|
type: "danger",
|
1174
1318
|
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
1175
1319
|
});
|
1176
|
-
trackUsage("
|
1320
|
+
trackUsage("didNotBulkDeleteEntries");
|
1177
1321
|
throw err;
|
1178
1322
|
}
|
1179
1323
|
},
|
1180
|
-
[trackUsage,
|
1324
|
+
[trackUsage, deleteManyDocuments, toggleNotification, formatMessage, formatAPIError]
|
1181
1325
|
);
|
1182
|
-
const [
|
1183
|
-
const
|
1184
|
-
async ({ model, params }
|
1326
|
+
const [discardDocument] = useDiscardDocumentMutation();
|
1327
|
+
const discard = React.useCallback(
|
1328
|
+
async ({ collectionType, model, documentId, params }) => {
|
1185
1329
|
try {
|
1186
|
-
const res = await
|
1330
|
+
const res = await discardDocument({
|
1331
|
+
collectionType,
|
1187
1332
|
model,
|
1188
|
-
|
1333
|
+
documentId,
|
1189
1334
|
params
|
1190
1335
|
});
|
1191
1336
|
if ("error" in res) {
|
1192
|
-
toggleNotification({
|
1193
|
-
|
1337
|
+
toggleNotification({
|
1338
|
+
type: "danger",
|
1339
|
+
message: formatAPIError(res.error)
|
1340
|
+
});
|
1194
1341
|
return { error: res.error };
|
1195
1342
|
}
|
1196
|
-
trackUsage("didCreateEntry", trackerProperty);
|
1197
1343
|
toggleNotification({
|
1198
1344
|
type: "success",
|
1199
1345
|
message: formatMessage({
|
1200
|
-
id:
|
1201
|
-
defaultMessage: "
|
1346
|
+
id: "content-manager.success.record.discard",
|
1347
|
+
defaultMessage: "Changes discarded"
|
1202
1348
|
})
|
1203
1349
|
});
|
1204
1350
|
return res.data;
|
@@ -1207,11 +1353,226 @@ const useDocumentActions = () => {
|
|
1207
1353
|
type: "danger",
|
1208
1354
|
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
1209
1355
|
});
|
1210
|
-
trackUsage("didNotCreateEntry", { error: err, ...trackerProperty });
|
1211
1356
|
throw err;
|
1212
1357
|
}
|
1213
1358
|
},
|
1214
|
-
[
|
1359
|
+
[discardDocument, formatAPIError, formatMessage, toggleNotification]
|
1360
|
+
);
|
1361
|
+
const [publishDocument] = usePublishDocumentMutation();
|
1362
|
+
const publish = React.useCallback(
|
1363
|
+
async ({ collectionType, model, documentId, params }, data) => {
|
1364
|
+
try {
|
1365
|
+
trackUsage("willPublishEntry");
|
1366
|
+
const res = await publishDocument({
|
1367
|
+
collectionType,
|
1368
|
+
model,
|
1369
|
+
documentId,
|
1370
|
+
data,
|
1371
|
+
params
|
1372
|
+
});
|
1373
|
+
if ("error" in res) {
|
1374
|
+
toggleNotification({ type: "danger", message: formatAPIError(res.error) });
|
1375
|
+
return { error: res.error };
|
1376
|
+
}
|
1377
|
+
trackUsage("didPublishEntry");
|
1378
|
+
toggleNotification({
|
1379
|
+
type: "success",
|
1380
|
+
message: formatMessage({
|
1381
|
+
id: getTranslation("success.record.publish"),
|
1382
|
+
defaultMessage: "Published document"
|
1383
|
+
})
|
1384
|
+
});
|
1385
|
+
return res.data;
|
1386
|
+
} catch (err) {
|
1387
|
+
toggleNotification({
|
1388
|
+
type: "danger",
|
1389
|
+
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
1390
|
+
});
|
1391
|
+
throw err;
|
1392
|
+
}
|
1393
|
+
},
|
1394
|
+
[trackUsage, publishDocument, toggleNotification, formatMessage, formatAPIError]
|
1395
|
+
);
|
1396
|
+
const [publishManyDocuments] = usePublishManyDocumentsMutation();
|
1397
|
+
const publishMany = React.useCallback(
|
1398
|
+
async ({ model, documentIds, params }) => {
|
1399
|
+
try {
|
1400
|
+
const res = await publishManyDocuments({
|
1401
|
+
model,
|
1402
|
+
documentIds,
|
1403
|
+
params
|
1404
|
+
});
|
1405
|
+
if ("error" in res) {
|
1406
|
+
toggleNotification({ type: "danger", message: formatAPIError(res.error) });
|
1407
|
+
return { error: res.error };
|
1408
|
+
}
|
1409
|
+
toggleNotification({
|
1410
|
+
type: "success",
|
1411
|
+
message: formatMessage({
|
1412
|
+
id: getTranslation("success.record.publish"),
|
1413
|
+
defaultMessage: "Published document"
|
1414
|
+
})
|
1415
|
+
});
|
1416
|
+
return res.data;
|
1417
|
+
} catch (err) {
|
1418
|
+
toggleNotification({
|
1419
|
+
type: "danger",
|
1420
|
+
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
1421
|
+
});
|
1422
|
+
throw err;
|
1423
|
+
}
|
1424
|
+
},
|
1425
|
+
[
|
1426
|
+
// trackUsage,
|
1427
|
+
publishManyDocuments,
|
1428
|
+
toggleNotification,
|
1429
|
+
formatMessage,
|
1430
|
+
formatAPIError
|
1431
|
+
]
|
1432
|
+
);
|
1433
|
+
const [updateDocument] = useUpdateDocumentMutation();
|
1434
|
+
const update = React.useCallback(
|
1435
|
+
async ({ collectionType, model, documentId, params }, data, trackerProperty) => {
|
1436
|
+
try {
|
1437
|
+
trackUsage("willEditEntry", trackerProperty);
|
1438
|
+
const res = await updateDocument({
|
1439
|
+
collectionType,
|
1440
|
+
model,
|
1441
|
+
documentId,
|
1442
|
+
data,
|
1443
|
+
params
|
1444
|
+
});
|
1445
|
+
if ("error" in res) {
|
1446
|
+
toggleNotification({ type: "danger", message: formatAPIError(res.error) });
|
1447
|
+
trackUsage("didNotEditEntry", { error: res.error, ...trackerProperty });
|
1448
|
+
return { error: res.error };
|
1449
|
+
}
|
1450
|
+
trackUsage("didEditEntry", trackerProperty);
|
1451
|
+
toggleNotification({
|
1452
|
+
type: "success",
|
1453
|
+
message: formatMessage({
|
1454
|
+
id: getTranslation("success.record.save"),
|
1455
|
+
defaultMessage: "Saved document"
|
1456
|
+
})
|
1457
|
+
});
|
1458
|
+
return res.data;
|
1459
|
+
} catch (err) {
|
1460
|
+
trackUsage("didNotEditEntry", { error: err, ...trackerProperty });
|
1461
|
+
toggleNotification({
|
1462
|
+
type: "danger",
|
1463
|
+
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
1464
|
+
});
|
1465
|
+
throw err;
|
1466
|
+
}
|
1467
|
+
},
|
1468
|
+
[trackUsage, updateDocument, toggleNotification, formatMessage, formatAPIError]
|
1469
|
+
);
|
1470
|
+
const [unpublishDocument] = useUnpublishDocumentMutation();
|
1471
|
+
const unpublish = React.useCallback(
|
1472
|
+
async ({ collectionType, model, documentId, params }, discardDraft = false) => {
|
1473
|
+
try {
|
1474
|
+
trackUsage("willUnpublishEntry");
|
1475
|
+
const res = await unpublishDocument({
|
1476
|
+
collectionType,
|
1477
|
+
model,
|
1478
|
+
documentId,
|
1479
|
+
params,
|
1480
|
+
data: {
|
1481
|
+
discardDraft
|
1482
|
+
}
|
1483
|
+
});
|
1484
|
+
if ("error" in res) {
|
1485
|
+
toggleNotification({ type: "danger", message: formatAPIError(res.error) });
|
1486
|
+
return { error: res.error };
|
1487
|
+
}
|
1488
|
+
trackUsage("didUnpublishEntry");
|
1489
|
+
toggleNotification({
|
1490
|
+
type: "success",
|
1491
|
+
message: formatMessage({
|
1492
|
+
id: getTranslation("success.record.unpublish"),
|
1493
|
+
defaultMessage: "Unpublished document"
|
1494
|
+
})
|
1495
|
+
});
|
1496
|
+
return res.data;
|
1497
|
+
} catch (err) {
|
1498
|
+
toggleNotification({
|
1499
|
+
type: "danger",
|
1500
|
+
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
1501
|
+
});
|
1502
|
+
throw err;
|
1503
|
+
}
|
1504
|
+
},
|
1505
|
+
[trackUsage, unpublishDocument, toggleNotification, formatMessage, formatAPIError]
|
1506
|
+
);
|
1507
|
+
const [unpublishManyDocuments] = useUnpublishManyDocumentsMutation();
|
1508
|
+
const unpublishMany = React.useCallback(
|
1509
|
+
async ({ model, documentIds, params }) => {
|
1510
|
+
try {
|
1511
|
+
trackUsage("willBulkUnpublishEntries");
|
1512
|
+
const res = await unpublishManyDocuments({
|
1513
|
+
model,
|
1514
|
+
documentIds,
|
1515
|
+
params
|
1516
|
+
});
|
1517
|
+
if ("error" in res) {
|
1518
|
+
toggleNotification({ type: "danger", message: formatAPIError(res.error) });
|
1519
|
+
return { error: res.error };
|
1520
|
+
}
|
1521
|
+
trackUsage("didBulkUnpublishEntries");
|
1522
|
+
toggleNotification({
|
1523
|
+
type: "success",
|
1524
|
+
title: formatMessage({
|
1525
|
+
id: getTranslation("success.records.unpublish"),
|
1526
|
+
defaultMessage: "Successfully unpublished."
|
1527
|
+
}),
|
1528
|
+
message: ""
|
1529
|
+
});
|
1530
|
+
return res.data;
|
1531
|
+
} catch (err) {
|
1532
|
+
toggleNotification({
|
1533
|
+
type: "danger",
|
1534
|
+
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
1535
|
+
});
|
1536
|
+
trackUsage("didNotBulkUnpublishEntries");
|
1537
|
+
throw err;
|
1538
|
+
}
|
1539
|
+
},
|
1540
|
+
[trackUsage, unpublishManyDocuments, toggleNotification, formatMessage, formatAPIError]
|
1541
|
+
);
|
1542
|
+
const [createDocument] = useCreateDocumentMutation();
|
1543
|
+
const create = React.useCallback(
|
1544
|
+
async ({ model, params }, data, trackerProperty) => {
|
1545
|
+
try {
|
1546
|
+
const res = await createDocument({
|
1547
|
+
model,
|
1548
|
+
data,
|
1549
|
+
params
|
1550
|
+
});
|
1551
|
+
if ("error" in res) {
|
1552
|
+
toggleNotification({ type: "danger", message: formatAPIError(res.error) });
|
1553
|
+
trackUsage("didNotCreateEntry", { error: res.error, ...trackerProperty });
|
1554
|
+
return { error: res.error };
|
1555
|
+
}
|
1556
|
+
trackUsage("didCreateEntry", trackerProperty);
|
1557
|
+
toggleNotification({
|
1558
|
+
type: "success",
|
1559
|
+
message: formatMessage({
|
1560
|
+
id: getTranslation("success.record.save"),
|
1561
|
+
defaultMessage: "Saved document"
|
1562
|
+
})
|
1563
|
+
});
|
1564
|
+
setCurrentStep("contentManager.success");
|
1565
|
+
return res.data;
|
1566
|
+
} catch (err) {
|
1567
|
+
toggleNotification({
|
1568
|
+
type: "danger",
|
1569
|
+
message: formatMessage(DEFAULT_UNEXPECTED_ERROR_MSG)
|
1570
|
+
});
|
1571
|
+
trackUsage("didNotCreateEntry", { error: err, ...trackerProperty });
|
1572
|
+
throw err;
|
1573
|
+
}
|
1574
|
+
},
|
1575
|
+
[createDocument, formatAPIError, formatMessage, toggleNotification, trackUsage]
|
1215
1576
|
);
|
1216
1577
|
const [autoCloneDocument] = useAutoCloneDocumentMutation();
|
1217
1578
|
const autoClone = React.useCallback(
|
@@ -1303,7 +1664,7 @@ const useDocumentActions = () => {
|
|
1303
1664
|
};
|
1304
1665
|
};
|
1305
1666
|
const ProtectedHistoryPage = lazy(
|
1306
|
-
() => import("./History-
|
1667
|
+
() => import("./History-DlL3k_OW.mjs").then((mod) => ({ default: mod.ProtectedHistoryPage }))
|
1307
1668
|
);
|
1308
1669
|
const routes$1 = [
|
1309
1670
|
{
|
@@ -1316,31 +1677,31 @@ const routes$1 = [
|
|
1316
1677
|
}
|
1317
1678
|
];
|
1318
1679
|
const ProtectedEditViewPage = lazy(
|
1319
|
-
() => import("./EditViewPage-
|
1680
|
+
() => import("./EditViewPage-D5Zhyw_q.mjs").then((mod) => ({ default: mod.ProtectedEditViewPage }))
|
1320
1681
|
);
|
1321
1682
|
const ProtectedListViewPage = lazy(
|
1322
|
-
() => import("./ListViewPage-
|
1683
|
+
() => import("./ListViewPage-C9STpcYd.mjs").then((mod) => ({ default: mod.ProtectedListViewPage }))
|
1323
1684
|
);
|
1324
1685
|
const ProtectedListConfiguration = lazy(
|
1325
|
-
() => import("./ListConfigurationPage-
|
1686
|
+
() => import("./ListConfigurationPage-xj3cSF5M.mjs").then((mod) => ({
|
1326
1687
|
default: mod.ProtectedListConfiguration
|
1327
1688
|
}))
|
1328
1689
|
);
|
1329
1690
|
const ProtectedEditConfigurationPage = lazy(
|
1330
|
-
() => import("./EditConfigurationPage-
|
1691
|
+
() => import("./EditConfigurationPage-CBscKbxt.mjs").then((mod) => ({
|
1331
1692
|
default: mod.ProtectedEditConfigurationPage
|
1332
1693
|
}))
|
1333
1694
|
);
|
1334
1695
|
const ProtectedComponentConfigurationPage = lazy(
|
1335
|
-
() => import("./ComponentConfigurationPage-
|
1696
|
+
() => import("./ComponentConfigurationPage-suSMW3KC.mjs").then((mod) => ({
|
1336
1697
|
default: mod.ProtectedComponentConfigurationPage
|
1337
1698
|
}))
|
1338
1699
|
);
|
1339
1700
|
const NoPermissions = lazy(
|
1340
|
-
() => import("./NoPermissionsPage-
|
1701
|
+
() => import("./NoPermissionsPage-BeDixsE9.mjs").then((mod) => ({ default: mod.NoPermissions }))
|
1341
1702
|
);
|
1342
1703
|
const NoContentType = lazy(
|
1343
|
-
() => import("./NoContentTypePage-
|
1704
|
+
() => import("./NoContentTypePage-DpV_ouD6.mjs").then((mod) => ({ default: mod.NoContentType }))
|
1344
1705
|
);
|
1345
1706
|
const CollectionTypePages = () => {
|
1346
1707
|
const { collectionType } = useParams();
|
@@ -1397,1057 +1758,746 @@ const DocumentActions = ({ actions: actions2 }) => {
|
|
1397
1758
|
const positions = Array.isArray(action.position) ? action.position : [action.position];
|
1398
1759
|
return positions.includes("panel");
|
1399
1760
|
});
|
1400
|
-
if (!primaryAction) {
|
1401
|
-
return null;
|
1402
|
-
}
|
1403
|
-
return /* @__PURE__ */ jsxs(Flex, { direction: "column", gap: 2, alignItems: "stretch", width: "100%", children: [
|
1404
|
-
/* @__PURE__ */ jsxs(Flex, { gap: 2, children: [
|
1405
|
-
/* @__PURE__ */ jsx(DocumentActionButton, { ...primaryAction, variant: primaryAction.variant || "default" }),
|
1406
|
-
restActions.length > 0 ? /* @__PURE__ */ jsx(
|
1407
|
-
DocumentActionsMenu,
|
1408
|
-
{
|
1409
|
-
actions: restActions,
|
1410
|
-
label: formatMessage({
|
1411
|
-
id: "content-manager.containers.edit.panels.default.more-actions",
|
1412
|
-
defaultMessage: "More document actions"
|
1413
|
-
})
|
1414
|
-
}
|
1415
|
-
) : null
|
1416
|
-
] }),
|
1417
|
-
secondaryAction ? /* @__PURE__ */ jsx(
|
1418
|
-
DocumentActionButton,
|
1419
|
-
{
|
1420
|
-
...secondaryAction,
|
1421
|
-
variant: secondaryAction.variant || "secondary"
|
1422
|
-
}
|
1423
|
-
) : null
|
1424
|
-
] });
|
1425
|
-
};
|
1426
|
-
const DocumentActionButton = (action) => {
|
1427
|
-
const [dialogId, setDialogId] = React.useState(null);
|
1428
|
-
const { toggleNotification } = useNotification();
|
1429
|
-
const handleClick = (action2) => async (e) => {
|
1430
|
-
const { onClick = () => false, dialog, id } = action2;
|
1431
|
-
const muteDialog = await onClick(e);
|
1432
|
-
if (dialog && !muteDialog) {
|
1433
|
-
switch (dialog.type) {
|
1434
|
-
case "notification":
|
1435
|
-
toggleNotification({
|
1436
|
-
title: dialog.title,
|
1437
|
-
message: dialog.content,
|
1438
|
-
type: dialog.status,
|
1439
|
-
timeout: dialog.timeout,
|
1440
|
-
onClose: dialog.onClose
|
1441
|
-
});
|
1442
|
-
break;
|
1443
|
-
case "dialog":
|
1444
|
-
case "modal":
|
1445
|
-
e.preventDefault();
|
1446
|
-
setDialogId(id);
|
1447
|
-
}
|
1448
|
-
}
|
1449
|
-
};
|
1450
|
-
const handleClose = () => {
|
1451
|
-
setDialogId(null);
|
1452
|
-
};
|
1453
|
-
return /* @__PURE__ */ jsxs(Fragment, { children: [
|
1454
|
-
/* @__PURE__ */ jsx(
|
1455
|
-
Button,
|
1456
|
-
{
|
1457
|
-
flex: "auto",
|
1458
|
-
startIcon: action.icon,
|
1459
|
-
disabled: action.disabled,
|
1460
|
-
onClick: handleClick(action),
|
1461
|
-
justifyContent: "center",
|
1462
|
-
variant: action.variant || "default",
|
1463
|
-
paddingTop: "7px",
|
1464
|
-
paddingBottom: "7px",
|
1465
|
-
children: action.label
|
1466
|
-
}
|
1467
|
-
),
|
1468
|
-
action.dialog?.type === "dialog" ? /* @__PURE__ */ jsx(
|
1469
|
-
DocumentActionConfirmDialog,
|
1470
|
-
{
|
1471
|
-
...action.dialog,
|
1472
|
-
variant: action.dialog?.variant ?? action.variant,
|
1473
|
-
isOpen: dialogId === action.id,
|
1474
|
-
onClose: handleClose
|
1475
|
-
}
|
1476
|
-
) : null,
|
1477
|
-
action.dialog?.type === "modal" ? /* @__PURE__ */ jsx(
|
1478
|
-
DocumentActionModal,
|
1479
|
-
{
|
1480
|
-
...action.dialog,
|
1481
|
-
onModalClose: handleClose,
|
1482
|
-
isOpen: dialogId === action.id
|
1483
|
-
}
|
1484
|
-
) : null
|
1485
|
-
] });
|
1486
|
-
};
|
1487
|
-
const DocumentActionsMenu = ({
|
1488
|
-
actions: actions2,
|
1489
|
-
children,
|
1490
|
-
label,
|
1491
|
-
variant = "tertiary"
|
1492
|
-
}) => {
|
1493
|
-
const [isOpen, setIsOpen] = React.useState(false);
|
1494
|
-
const [dialogId, setDialogId] = React.useState(null);
|
1495
|
-
const { formatMessage } = useIntl();
|
1496
|
-
const { toggleNotification } = useNotification();
|
1497
|
-
const isDisabled = actions2.every((action) => action.disabled) || actions2.length === 0;
|
1498
|
-
const handleClick = (action) => async (e) => {
|
1499
|
-
const { onClick = () => false, dialog, id } = action;
|
1500
|
-
const muteDialog = await onClick(e);
|
1501
|
-
if (dialog && !muteDialog) {
|
1502
|
-
switch (dialog.type) {
|
1503
|
-
case "notification":
|
1504
|
-
toggleNotification({
|
1505
|
-
title: dialog.title,
|
1506
|
-
message: dialog.content,
|
1507
|
-
type: dialog.status,
|
1508
|
-
timeout: dialog.timeout,
|
1509
|
-
onClose: dialog.onClose
|
1510
|
-
});
|
1511
|
-
break;
|
1512
|
-
case "dialog":
|
1513
|
-
case "modal":
|
1514
|
-
setDialogId(id);
|
1515
|
-
}
|
1516
|
-
}
|
1517
|
-
};
|
1518
|
-
const handleClose = () => {
|
1519
|
-
setDialogId(null);
|
1520
|
-
setIsOpen(false);
|
1521
|
-
};
|
1522
|
-
return /* @__PURE__ */ jsxs(Menu.Root, { open: isOpen, onOpenChange: setIsOpen, children: [
|
1523
|
-
/* @__PURE__ */ jsxs(
|
1524
|
-
Menu.Trigger,
|
1525
|
-
{
|
1526
|
-
disabled: isDisabled,
|
1527
|
-
size: "S",
|
1528
|
-
endIcon: null,
|
1529
|
-
paddingTop: "4px",
|
1530
|
-
paddingLeft: "7px",
|
1531
|
-
paddingRight: "7px",
|
1532
|
-
variant,
|
1533
|
-
children: [
|
1534
|
-
/* @__PURE__ */ jsx(More, { "aria-hidden": true, focusable: false }),
|
1535
|
-
/* @__PURE__ */ jsx(VisuallyHidden, { tag: "span", children: label || formatMessage({
|
1536
|
-
id: "content-manager.containers.edit.panels.default.more-actions",
|
1537
|
-
defaultMessage: "More document actions"
|
1538
|
-
}) })
|
1539
|
-
]
|
1540
|
-
}
|
1541
|
-
),
|
1542
|
-
/* @__PURE__ */ jsxs(Menu.Content, { top: "4px", maxHeight: void 0, popoverPlacement: "bottom-end", children: [
|
1543
|
-
actions2.map((action) => {
|
1544
|
-
return /* @__PURE__ */ jsx(
|
1545
|
-
Menu.Item,
|
1546
|
-
{
|
1547
|
-
disabled: action.disabled,
|
1548
|
-
onSelect: handleClick(action),
|
1549
|
-
display: "block",
|
1550
|
-
children: /* @__PURE__ */ jsxs(Flex, { justifyContent: "space-between", gap: 4, children: [
|
1551
|
-
/* @__PURE__ */ jsxs(
|
1552
|
-
Flex,
|
1553
|
-
{
|
1554
|
-
color: !action.disabled ? convertActionVariantToColor(action.variant) : "inherit",
|
1555
|
-
gap: 2,
|
1556
|
-
tag: "span",
|
1557
|
-
children: [
|
1558
|
-
/* @__PURE__ */ jsx(
|
1559
|
-
Flex,
|
1560
|
-
{
|
1561
|
-
tag: "span",
|
1562
|
-
color: !action.disabled ? convertActionVariantToIconColor(action.variant) : "inherit",
|
1563
|
-
children: action.icon
|
1564
|
-
}
|
1565
|
-
),
|
1566
|
-
action.label
|
1567
|
-
]
|
1568
|
-
}
|
1569
|
-
),
|
1570
|
-
action.id.startsWith("HistoryAction") && /* @__PURE__ */ jsx(
|
1571
|
-
Flex,
|
1572
|
-
{
|
1573
|
-
alignItems: "center",
|
1574
|
-
background: "alternative100",
|
1575
|
-
borderStyle: "solid",
|
1576
|
-
borderColor: "alternative200",
|
1577
|
-
borderWidth: "1px",
|
1578
|
-
height: 5,
|
1579
|
-
paddingLeft: 2,
|
1580
|
-
paddingRight: 2,
|
1581
|
-
hasRadius: true,
|
1582
|
-
color: "alternative600",
|
1583
|
-
children: /* @__PURE__ */ jsx(Typography, { variant: "sigma", fontWeight: "bold", lineHeight: 1, children: formatMessage({ id: "global.new", defaultMessage: "New" }) })
|
1584
|
-
}
|
1585
|
-
)
|
1586
|
-
] })
|
1587
|
-
},
|
1588
|
-
action.id
|
1589
|
-
);
|
1590
|
-
}),
|
1591
|
-
children
|
1592
|
-
] }),
|
1593
|
-
actions2.map((action) => {
|
1594
|
-
return /* @__PURE__ */ jsxs(React.Fragment, { children: [
|
1595
|
-
action.dialog?.type === "dialog" ? /* @__PURE__ */ jsx(
|
1596
|
-
DocumentActionConfirmDialog,
|
1597
|
-
{
|
1598
|
-
...action.dialog,
|
1599
|
-
variant: action.variant,
|
1600
|
-
isOpen: dialogId === action.id,
|
1601
|
-
onClose: handleClose
|
1602
|
-
}
|
1603
|
-
) : null,
|
1604
|
-
action.dialog?.type === "modal" ? /* @__PURE__ */ jsx(
|
1605
|
-
DocumentActionModal,
|
1606
|
-
{
|
1607
|
-
...action.dialog,
|
1608
|
-
onModalClose: handleClose,
|
1609
|
-
isOpen: dialogId === action.id
|
1610
|
-
}
|
1611
|
-
) : null
|
1612
|
-
] }, action.id);
|
1613
|
-
})
|
1614
|
-
] });
|
1615
|
-
};
|
1616
|
-
const convertActionVariantToColor = (variant = "secondary") => {
|
1617
|
-
switch (variant) {
|
1618
|
-
case "danger":
|
1619
|
-
return "danger600";
|
1620
|
-
case "secondary":
|
1621
|
-
return void 0;
|
1622
|
-
case "success":
|
1623
|
-
return "success600";
|
1624
|
-
default:
|
1625
|
-
return "primary600";
|
1626
|
-
}
|
1627
|
-
};
|
1628
|
-
const convertActionVariantToIconColor = (variant = "secondary") => {
|
1629
|
-
switch (variant) {
|
1630
|
-
case "danger":
|
1631
|
-
return "danger600";
|
1632
|
-
case "secondary":
|
1633
|
-
return "neutral500";
|
1634
|
-
case "success":
|
1635
|
-
return "success600";
|
1636
|
-
default:
|
1637
|
-
return "primary600";
|
1638
|
-
}
|
1639
|
-
};
|
1640
|
-
const DocumentActionConfirmDialog = ({
|
1641
|
-
onClose,
|
1642
|
-
onCancel,
|
1643
|
-
onConfirm,
|
1644
|
-
title,
|
1645
|
-
content,
|
1646
|
-
isOpen,
|
1647
|
-
variant = "secondary"
|
1648
|
-
}) => {
|
1649
|
-
const { formatMessage } = useIntl();
|
1650
|
-
const handleClose = async () => {
|
1651
|
-
if (onCancel) {
|
1652
|
-
await onCancel();
|
1653
|
-
}
|
1654
|
-
onClose();
|
1655
|
-
};
|
1656
|
-
const handleConfirm = async () => {
|
1657
|
-
if (onConfirm) {
|
1658
|
-
await onConfirm();
|
1659
|
-
}
|
1660
|
-
onClose();
|
1661
|
-
};
|
1662
|
-
return /* @__PURE__ */ jsx(Dialog.Root, { open: isOpen, onOpenChange: handleClose, children: /* @__PURE__ */ jsxs(Dialog.Content, { children: [
|
1663
|
-
/* @__PURE__ */ jsx(Dialog.Header, { children: title }),
|
1664
|
-
/* @__PURE__ */ jsx(Dialog.Body, { children: content }),
|
1665
|
-
/* @__PURE__ */ jsxs(Dialog.Footer, { children: [
|
1666
|
-
/* @__PURE__ */ jsx(Dialog.Cancel, { children: /* @__PURE__ */ jsx(Button, { variant: "tertiary", children: formatMessage({
|
1667
|
-
id: "app.components.Button.cancel",
|
1668
|
-
defaultMessage: "Cancel"
|
1669
|
-
}) }) }),
|
1670
|
-
/* @__PURE__ */ jsx(Button, { onClick: handleConfirm, variant, children: formatMessage({
|
1671
|
-
id: "app.components.Button.confirm",
|
1672
|
-
defaultMessage: "Confirm"
|
1673
|
-
}) })
|
1674
|
-
] })
|
1675
|
-
] }) });
|
1676
|
-
};
|
1677
|
-
const DocumentActionModal = ({
|
1678
|
-
isOpen,
|
1679
|
-
title,
|
1680
|
-
onClose,
|
1681
|
-
footer: Footer,
|
1682
|
-
content: Content,
|
1683
|
-
onModalClose
|
1684
|
-
}) => {
|
1685
|
-
const handleClose = () => {
|
1686
|
-
if (onClose) {
|
1687
|
-
onClose();
|
1688
|
-
}
|
1689
|
-
onModalClose();
|
1690
|
-
};
|
1691
|
-
return /* @__PURE__ */ jsx(Modal.Root, { open: isOpen, onOpenChange: handleClose, children: /* @__PURE__ */ jsxs(Modal.Content, { children: [
|
1692
|
-
/* @__PURE__ */ jsx(Modal.Header, { children: /* @__PURE__ */ jsx(Modal.Title, { children: title }) }),
|
1693
|
-
typeof Content === "function" ? /* @__PURE__ */ jsx(Content, { onClose: handleClose }) : /* @__PURE__ */ jsx(Modal.Body, { children: Content }),
|
1694
|
-
typeof Footer === "function" ? /* @__PURE__ */ jsx(Footer, { onClose: handleClose }) : Footer
|
1695
|
-
] }) });
|
1696
|
-
};
|
1697
|
-
const PublishAction$1 = ({
|
1698
|
-
activeTab,
|
1699
|
-
documentId,
|
1700
|
-
model,
|
1701
|
-
collectionType,
|
1702
|
-
meta,
|
1703
|
-
document
|
1704
|
-
}) => {
|
1705
|
-
const { schema } = useDoc();
|
1706
|
-
const navigate = useNavigate();
|
1707
|
-
const { toggleNotification } = useNotification();
|
1708
|
-
const { _unstableFormatValidationErrors: formatValidationErrors } = useAPIErrorHandler();
|
1709
|
-
const isListView = useMatch(LIST_PATH) !== null;
|
1710
|
-
const isCloning = useMatch(CLONE_PATH) !== null;
|
1711
|
-
const { formatMessage } = useIntl();
|
1712
|
-
const canPublish = useDocumentRBAC("PublishAction", ({ canPublish: canPublish2 }) => canPublish2);
|
1713
|
-
const { publish } = useDocumentActions();
|
1714
|
-
const [
|
1715
|
-
countDraftRelations,
|
1716
|
-
{ isLoading: isLoadingDraftRelations, isError: isErrorDraftRelations }
|
1717
|
-
] = useLazyGetDraftRelationCountQuery();
|
1718
|
-
const [localCountOfDraftRelations, setLocalCountOfDraftRelations] = React.useState(0);
|
1719
|
-
const [serverCountOfDraftRelations, setServerCountOfDraftRelations] = React.useState(0);
|
1720
|
-
const [{ query, rawQuery }] = useQueryParams();
|
1721
|
-
const params = React.useMemo(() => buildValidParams(query), [query]);
|
1722
|
-
const modified = useForm("PublishAction", ({ modified: modified2 }) => modified2);
|
1723
|
-
const setSubmitting = useForm("PublishAction", ({ setSubmitting: setSubmitting2 }) => setSubmitting2);
|
1724
|
-
const isSubmitting = useForm("PublishAction", ({ isSubmitting: isSubmitting2 }) => isSubmitting2);
|
1725
|
-
const validate = useForm("PublishAction", (state) => state.validate);
|
1726
|
-
const setErrors = useForm("PublishAction", (state) => state.setErrors);
|
1727
|
-
const formValues = useForm("PublishAction", ({ values }) => values);
|
1728
|
-
React.useEffect(() => {
|
1729
|
-
if (isErrorDraftRelations) {
|
1730
|
-
toggleNotification({
|
1731
|
-
type: "danger",
|
1732
|
-
message: formatMessage({
|
1733
|
-
id: getTranslation("error.records.fetch-draft-relatons"),
|
1734
|
-
defaultMessage: "An error occurred while fetching draft relations on this document."
|
1735
|
-
})
|
1736
|
-
});
|
1737
|
-
}
|
1738
|
-
}, [isErrorDraftRelations, toggleNotification, formatMessage]);
|
1739
|
-
React.useEffect(() => {
|
1740
|
-
const localDraftRelations = /* @__PURE__ */ new Set();
|
1741
|
-
const extractDraftRelations = (data) => {
|
1742
|
-
const relations = data.connect || [];
|
1743
|
-
relations.forEach((relation) => {
|
1744
|
-
if (relation.status === "draft") {
|
1745
|
-
localDraftRelations.add(relation.id);
|
1746
|
-
}
|
1747
|
-
});
|
1748
|
-
};
|
1749
|
-
const traverseAndExtract = (data) => {
|
1750
|
-
Object.entries(data).forEach(([key, value]) => {
|
1751
|
-
if (key === "connect" && Array.isArray(value)) {
|
1752
|
-
extractDraftRelations({ connect: value });
|
1753
|
-
} else if (typeof value === "object" && value !== null) {
|
1754
|
-
traverseAndExtract(value);
|
1755
|
-
}
|
1756
|
-
});
|
1757
|
-
};
|
1758
|
-
if (!documentId || modified) {
|
1759
|
-
traverseAndExtract(formValues);
|
1760
|
-
setLocalCountOfDraftRelations(localDraftRelations.size);
|
1761
|
-
}
|
1762
|
-
}, [documentId, modified, formValues, setLocalCountOfDraftRelations]);
|
1763
|
-
React.useEffect(() => {
|
1764
|
-
if (documentId && !isListView) {
|
1765
|
-
const fetchDraftRelationsCount = async () => {
|
1766
|
-
const { data, error } = await countDraftRelations({
|
1767
|
-
collectionType,
|
1768
|
-
model,
|
1769
|
-
documentId,
|
1770
|
-
params
|
1771
|
-
});
|
1772
|
-
if (error) {
|
1773
|
-
throw error;
|
1774
|
-
}
|
1775
|
-
if (data) {
|
1776
|
-
setServerCountOfDraftRelations(data.data);
|
1777
|
-
}
|
1778
|
-
};
|
1779
|
-
fetchDraftRelationsCount();
|
1780
|
-
}
|
1781
|
-
}, [isListView, documentId, countDraftRelations, collectionType, model, params]);
|
1782
|
-
const isDocumentPublished = (document?.[PUBLISHED_AT_ATTRIBUTE_NAME] || meta?.availableStatus.some((doc) => doc[PUBLISHED_AT_ATTRIBUTE_NAME] !== null)) && document?.status !== "modified";
|
1783
|
-
if (!schema?.options?.draftAndPublish) {
|
1761
|
+
if (!primaryAction) {
|
1784
1762
|
return null;
|
1785
1763
|
}
|
1786
|
-
|
1787
|
-
|
1788
|
-
|
1789
|
-
|
1790
|
-
|
1791
|
-
toggleNotification({
|
1792
|
-
type: "danger",
|
1793
|
-
message: formatMessage({
|
1794
|
-
id: "content-manager.validation.error",
|
1795
|
-
defaultMessage: "There are validation errors in your document. Please fix them before saving."
|
1796
|
-
})
|
1797
|
-
});
|
1798
|
-
return;
|
1799
|
-
}
|
1800
|
-
const res = await publish(
|
1801
|
-
{
|
1802
|
-
collectionType,
|
1803
|
-
model,
|
1804
|
-
documentId,
|
1805
|
-
params
|
1806
|
-
},
|
1807
|
-
formValues
|
1808
|
-
);
|
1809
|
-
if ("data" in res && collectionType !== SINGLE_TYPES) {
|
1810
|
-
navigate({
|
1811
|
-
pathname: `../${collectionType}/${model}/${res.data.documentId}`,
|
1812
|
-
search: rawQuery
|
1813
|
-
});
|
1814
|
-
} else if ("error" in res && isBaseQueryError(res.error) && res.error.name === "ValidationError") {
|
1815
|
-
setErrors(formatValidationErrors(res.error));
|
1816
|
-
}
|
1817
|
-
} finally {
|
1818
|
-
setSubmitting(false);
|
1819
|
-
}
|
1820
|
-
};
|
1821
|
-
const totalDraftRelations = localCountOfDraftRelations + serverCountOfDraftRelations;
|
1822
|
-
const enableDraftRelationsCount = false;
|
1823
|
-
const hasDraftRelations = enableDraftRelationsCount;
|
1824
|
-
return {
|
1825
|
-
/**
|
1826
|
-
* Disabled when:
|
1827
|
-
* - currently if you're cloning a document we don't support publish & clone at the same time.
|
1828
|
-
* - the form is submitting
|
1829
|
-
* - the active tab is the published tab
|
1830
|
-
* - the document is already published & not modified
|
1831
|
-
* - the document is being created & not modified
|
1832
|
-
* - the user doesn't have the permission to publish
|
1833
|
-
*/
|
1834
|
-
disabled: isCloning || isSubmitting || isLoadingDraftRelations || activeTab === "published" || !modified && isDocumentPublished || !modified && !document?.documentId || !canPublish,
|
1835
|
-
label: formatMessage({
|
1836
|
-
id: "app.utils.publish",
|
1837
|
-
defaultMessage: "Publish"
|
1838
|
-
}),
|
1839
|
-
onClick: async () => {
|
1840
|
-
await performPublish();
|
1841
|
-
},
|
1842
|
-
dialog: hasDraftRelations ? {
|
1843
|
-
type: "dialog",
|
1844
|
-
variant: "danger",
|
1845
|
-
footer: null,
|
1846
|
-
title: formatMessage({
|
1847
|
-
id: getTranslation(`popUpwarning.warning.bulk-has-draft-relations.title`),
|
1848
|
-
defaultMessage: "Confirmation"
|
1849
|
-
}),
|
1850
|
-
content: formatMessage(
|
1851
|
-
{
|
1852
|
-
id: getTranslation(`popUpwarning.warning.bulk-has-draft-relations.message`),
|
1853
|
-
defaultMessage: "This entry is related to {count, plural, one {# draft entry} other {# draft entries}}. Publishing it could leave broken links in your app."
|
1854
|
-
},
|
1764
|
+
return /* @__PURE__ */ jsxs(Flex, { direction: "column", gap: 2, alignItems: "stretch", width: "100%", children: [
|
1765
|
+
/* @__PURE__ */ jsxs(Flex, { gap: 2, children: [
|
1766
|
+
/* @__PURE__ */ jsx(DocumentActionButton, { ...primaryAction, variant: primaryAction.variant || "default" }),
|
1767
|
+
restActions.length > 0 ? /* @__PURE__ */ jsx(
|
1768
|
+
DocumentActionsMenu,
|
1855
1769
|
{
|
1856
|
-
|
1770
|
+
actions: restActions,
|
1771
|
+
label: formatMessage({
|
1772
|
+
id: "content-manager.containers.edit.panels.default.more-actions",
|
1773
|
+
defaultMessage: "More document actions"
|
1774
|
+
})
|
1857
1775
|
}
|
1858
|
-
)
|
1859
|
-
|
1860
|
-
|
1776
|
+
) : null
|
1777
|
+
] }),
|
1778
|
+
secondaryAction ? /* @__PURE__ */ jsx(
|
1779
|
+
DocumentActionButton,
|
1780
|
+
{
|
1781
|
+
...secondaryAction,
|
1782
|
+
variant: secondaryAction.variant || "secondary"
|
1861
1783
|
}
|
1862
|
-
|
1863
|
-
};
|
1784
|
+
) : null
|
1785
|
+
] });
|
1864
1786
|
};
|
1865
|
-
|
1866
|
-
const
|
1867
|
-
activeTab,
|
1868
|
-
documentId,
|
1869
|
-
model,
|
1870
|
-
collectionType
|
1871
|
-
}) => {
|
1872
|
-
const navigate = useNavigate();
|
1787
|
+
const DocumentActionButton = (action) => {
|
1788
|
+
const [dialogId, setDialogId] = React.useState(null);
|
1873
1789
|
const { toggleNotification } = useNotification();
|
1874
|
-
const
|
1875
|
-
|
1876
|
-
|
1877
|
-
|
1878
|
-
|
1879
|
-
|
1880
|
-
|
1881
|
-
|
1882
|
-
|
1883
|
-
|
1884
|
-
|
1885
|
-
|
1886
|
-
|
1887
|
-
|
1888
|
-
|
1889
|
-
|
1890
|
-
|
1891
|
-
|
1892
|
-
* - the document is not modified & we're not cloning (you can save a clone entity straight away)
|
1893
|
-
* - the active tab is the published tab
|
1894
|
-
*/
|
1895
|
-
disabled: isSubmitting || !modified && !isCloning || activeTab === "published",
|
1896
|
-
label: formatMessage({
|
1897
|
-
id: "content-manager.containers.Edit.save",
|
1898
|
-
defaultMessage: "Save"
|
1899
|
-
}),
|
1900
|
-
onClick: async () => {
|
1901
|
-
setSubmitting(true);
|
1902
|
-
try {
|
1903
|
-
if (activeTab !== "draft") {
|
1904
|
-
const { errors } = await validate();
|
1905
|
-
if (errors) {
|
1906
|
-
toggleNotification({
|
1907
|
-
type: "danger",
|
1908
|
-
message: formatMessage({
|
1909
|
-
id: "content-manager.validation.error",
|
1910
|
-
defaultMessage: "There are validation errors in your document. Please fix them before saving."
|
1911
|
-
})
|
1912
|
-
});
|
1913
|
-
return;
|
1914
|
-
}
|
1915
|
-
}
|
1916
|
-
if (isCloning) {
|
1917
|
-
const res = await clone(
|
1918
|
-
{
|
1919
|
-
model,
|
1920
|
-
documentId: cloneMatch.params.origin,
|
1921
|
-
params
|
1922
|
-
},
|
1923
|
-
document
|
1924
|
-
);
|
1925
|
-
if ("data" in res) {
|
1926
|
-
navigate(
|
1927
|
-
{
|
1928
|
-
pathname: `../${res.data.documentId}`,
|
1929
|
-
search: rawQuery
|
1930
|
-
},
|
1931
|
-
{ relative: "path" }
|
1932
|
-
);
|
1933
|
-
} else if ("error" in res && isBaseQueryError(res.error) && res.error.name === "ValidationError") {
|
1934
|
-
setErrors(formatValidationErrors(res.error));
|
1935
|
-
}
|
1936
|
-
} else if (documentId || collectionType === SINGLE_TYPES) {
|
1937
|
-
const res = await update(
|
1938
|
-
{
|
1939
|
-
collectionType,
|
1940
|
-
model,
|
1941
|
-
documentId,
|
1942
|
-
params
|
1943
|
-
},
|
1944
|
-
document
|
1945
|
-
);
|
1946
|
-
if ("error" in res && isBaseQueryError(res.error) && res.error.name === "ValidationError") {
|
1947
|
-
setErrors(formatValidationErrors(res.error));
|
1948
|
-
} else {
|
1949
|
-
resetForm();
|
1950
|
-
}
|
1951
|
-
} else {
|
1952
|
-
const res = await create(
|
1953
|
-
{
|
1954
|
-
model,
|
1955
|
-
params
|
1956
|
-
},
|
1957
|
-
document
|
1958
|
-
);
|
1959
|
-
if ("data" in res && collectionType !== SINGLE_TYPES) {
|
1960
|
-
navigate(
|
1961
|
-
{
|
1962
|
-
pathname: `../${res.data.documentId}`,
|
1963
|
-
search: rawQuery
|
1964
|
-
},
|
1965
|
-
{ replace: true, relative: "path" }
|
1966
|
-
);
|
1967
|
-
} else if ("error" in res && isBaseQueryError(res.error) && res.error.name === "ValidationError") {
|
1968
|
-
setErrors(formatValidationErrors(res.error));
|
1969
|
-
}
|
1970
|
-
}
|
1971
|
-
} finally {
|
1972
|
-
setSubmitting(false);
|
1790
|
+
const handleClick = (action2) => async (e) => {
|
1791
|
+
const { onClick = () => false, dialog, id } = action2;
|
1792
|
+
const muteDialog = await onClick(e);
|
1793
|
+
if (dialog && !muteDialog) {
|
1794
|
+
switch (dialog.type) {
|
1795
|
+
case "notification":
|
1796
|
+
toggleNotification({
|
1797
|
+
title: dialog.title,
|
1798
|
+
message: dialog.content,
|
1799
|
+
type: dialog.status,
|
1800
|
+
timeout: dialog.timeout,
|
1801
|
+
onClose: dialog.onClose
|
1802
|
+
});
|
1803
|
+
break;
|
1804
|
+
case "dialog":
|
1805
|
+
case "modal":
|
1806
|
+
e.preventDefault();
|
1807
|
+
setDialogId(id);
|
1973
1808
|
}
|
1974
1809
|
}
|
1975
1810
|
};
|
1976
|
-
|
1977
|
-
|
1978
|
-
|
1979
|
-
|
1980
|
-
|
1981
|
-
|
1982
|
-
|
1983
|
-
|
1984
|
-
|
1985
|
-
|
1986
|
-
|
1987
|
-
|
1811
|
+
const handleClose = () => {
|
1812
|
+
setDialogId(null);
|
1813
|
+
};
|
1814
|
+
return /* @__PURE__ */ jsxs(Fragment, { children: [
|
1815
|
+
/* @__PURE__ */ jsx(
|
1816
|
+
Button,
|
1817
|
+
{
|
1818
|
+
flex: "auto",
|
1819
|
+
startIcon: action.icon,
|
1820
|
+
disabled: action.disabled,
|
1821
|
+
onClick: handleClick(action),
|
1822
|
+
justifyContent: "center",
|
1823
|
+
variant: action.variant || "default",
|
1824
|
+
paddingTop: "7px",
|
1825
|
+
paddingBottom: "7px",
|
1826
|
+
children: action.label
|
1827
|
+
}
|
1828
|
+
),
|
1829
|
+
action.dialog?.type === "dialog" ? /* @__PURE__ */ jsx(
|
1830
|
+
DocumentActionConfirmDialog,
|
1831
|
+
{
|
1832
|
+
...action.dialog,
|
1833
|
+
variant: action.dialog?.variant ?? action.variant,
|
1834
|
+
isOpen: dialogId === action.id,
|
1835
|
+
onClose: handleClose
|
1836
|
+
}
|
1837
|
+
) : null,
|
1838
|
+
action.dialog?.type === "modal" ? /* @__PURE__ */ jsx(
|
1839
|
+
DocumentActionModal,
|
1840
|
+
{
|
1841
|
+
...action.dialog,
|
1842
|
+
onModalClose: handleClose,
|
1843
|
+
isOpen: dialogId === action.id
|
1844
|
+
}
|
1845
|
+
) : null
|
1846
|
+
] });
|
1847
|
+
};
|
1848
|
+
const DocumentActionsMenu = ({
|
1849
|
+
actions: actions2,
|
1850
|
+
children,
|
1851
|
+
label,
|
1852
|
+
variant = "tertiary"
|
1988
1853
|
}) => {
|
1854
|
+
const [isOpen, setIsOpen] = React.useState(false);
|
1855
|
+
const [dialogId, setDialogId] = React.useState(null);
|
1989
1856
|
const { formatMessage } = useIntl();
|
1990
|
-
const { schema } = useDoc();
|
1991
|
-
const canPublish = useDocumentRBAC("UnpublishAction", ({ canPublish: canPublish2 }) => canPublish2);
|
1992
|
-
const { unpublish } = useDocumentActions();
|
1993
|
-
const [{ query }] = useQueryParams();
|
1994
|
-
const params = React.useMemo(() => buildValidParams(query), [query]);
|
1995
1857
|
const { toggleNotification } = useNotification();
|
1996
|
-
const
|
1997
|
-
const
|
1998
|
-
|
1999
|
-
|
2000
|
-
|
2001
|
-
|
2002
|
-
|
2003
|
-
}
|
2004
|
-
return {
|
2005
|
-
disabled: !canPublish || activeTab === "published" || document?.status !== "published" && document?.status !== "modified",
|
2006
|
-
label: formatMessage({
|
2007
|
-
id: "app.utils.unpublish",
|
2008
|
-
defaultMessage: "Unpublish"
|
2009
|
-
}),
|
2010
|
-
icon: /* @__PURE__ */ jsx(StyledCrossCircle, {}),
|
2011
|
-
onClick: async () => {
|
2012
|
-
if (!documentId && collectionType !== SINGLE_TYPES || isDocumentModified) {
|
2013
|
-
if (!documentId) {
|
2014
|
-
console.error(
|
2015
|
-
"You're trying to unpublish a document without an id, this is likely a bug with Strapi. Please open an issue."
|
2016
|
-
);
|
1858
|
+
const isDisabled = actions2.every((action) => action.disabled) || actions2.length === 0;
|
1859
|
+
const handleClick = (action) => async (e) => {
|
1860
|
+
const { onClick = () => false, dialog, id } = action;
|
1861
|
+
const muteDialog = await onClick(e);
|
1862
|
+
if (dialog && !muteDialog) {
|
1863
|
+
switch (dialog.type) {
|
1864
|
+
case "notification":
|
2017
1865
|
toggleNotification({
|
2018
|
-
|
2019
|
-
|
2020
|
-
|
2021
|
-
|
2022
|
-
|
1866
|
+
title: dialog.title,
|
1867
|
+
message: dialog.content,
|
1868
|
+
type: dialog.status,
|
1869
|
+
timeout: dialog.timeout,
|
1870
|
+
onClose: dialog.onClose
|
2023
1871
|
});
|
2024
|
-
|
2025
|
-
|
1872
|
+
break;
|
1873
|
+
case "dialog":
|
1874
|
+
case "modal":
|
1875
|
+
setDialogId(id);
|
2026
1876
|
}
|
2027
|
-
|
2028
|
-
|
2029
|
-
|
2030
|
-
|
2031
|
-
|
2032
|
-
|
2033
|
-
|
2034
|
-
|
2035
|
-
|
2036
|
-
|
2037
|
-
|
2038
|
-
|
2039
|
-
|
2040
|
-
|
2041
|
-
|
2042
|
-
|
2043
|
-
|
2044
|
-
|
2045
|
-
|
1877
|
+
}
|
1878
|
+
};
|
1879
|
+
const handleClose = () => {
|
1880
|
+
setDialogId(null);
|
1881
|
+
setIsOpen(false);
|
1882
|
+
};
|
1883
|
+
return /* @__PURE__ */ jsxs(Menu.Root, { open: isOpen, onOpenChange: setIsOpen, children: [
|
1884
|
+
/* @__PURE__ */ jsxs(
|
1885
|
+
Menu.Trigger,
|
1886
|
+
{
|
1887
|
+
disabled: isDisabled,
|
1888
|
+
size: "S",
|
1889
|
+
endIcon: null,
|
1890
|
+
paddingTop: "4px",
|
1891
|
+
paddingLeft: "7px",
|
1892
|
+
paddingRight: "7px",
|
1893
|
+
variant,
|
1894
|
+
children: [
|
1895
|
+
/* @__PURE__ */ jsx(More, { "aria-hidden": true, focusable: false }),
|
1896
|
+
/* @__PURE__ */ jsx(VisuallyHidden, { tag: "span", children: label || formatMessage({
|
1897
|
+
id: "content-manager.containers.edit.panels.default.more-actions",
|
1898
|
+
defaultMessage: "More document actions"
|
2046
1899
|
}) })
|
2047
|
-
]
|
2048
|
-
|
2049
|
-
|
2050
|
-
|
2051
|
-
|
2052
|
-
|
2053
|
-
|
2054
|
-
id: "content-manager.actions.unpublish.dialog.radio-label",
|
2055
|
-
defaultMessage: "Choose an option to unpublish the document."
|
2056
|
-
}),
|
2057
|
-
onValueChange: handleChange,
|
2058
|
-
children: [
|
2059
|
-
/* @__PURE__ */ jsx(Radio.Item, { checked: shouldKeepDraft, value: UNPUBLISH_DRAFT_OPTIONS.KEEP, children: formatMessage({
|
2060
|
-
id: "content-manager.actions.unpublish.dialog.option.keep-draft",
|
2061
|
-
defaultMessage: "Keep draft"
|
2062
|
-
}) }),
|
2063
|
-
/* @__PURE__ */ jsx(Radio.Item, { checked: !shouldKeepDraft, value: UNPUBLISH_DRAFT_OPTIONS.DISCARD, children: formatMessage({
|
2064
|
-
id: "content-manager.actions.unpublish.dialog.option.replace-draft",
|
2065
|
-
defaultMessage: "Replace draft"
|
2066
|
-
}) })
|
2067
|
-
]
|
2068
|
-
}
|
2069
|
-
)
|
2070
|
-
] }),
|
2071
|
-
onConfirm: async () => {
|
2072
|
-
if (!documentId && collectionType !== SINGLE_TYPES) {
|
2073
|
-
console.error(
|
2074
|
-
"You're trying to unpublish a document without an id, this is likely a bug with Strapi. Please open an issue."
|
2075
|
-
);
|
2076
|
-
toggleNotification({
|
2077
|
-
message: formatMessage({
|
2078
|
-
id: "content-manager.actions.unpublish.error",
|
2079
|
-
defaultMessage: "An error occurred while trying to unpublish the document."
|
2080
|
-
}),
|
2081
|
-
type: "danger"
|
2082
|
-
});
|
2083
|
-
}
|
2084
|
-
await unpublish(
|
1900
|
+
]
|
1901
|
+
}
|
1902
|
+
),
|
1903
|
+
/* @__PURE__ */ jsxs(Menu.Content, { maxHeight: void 0, popoverPlacement: "bottom-end", children: [
|
1904
|
+
actions2.map((action) => {
|
1905
|
+
return /* @__PURE__ */ jsx(
|
1906
|
+
Menu.Item,
|
2085
1907
|
{
|
2086
|
-
|
2087
|
-
|
2088
|
-
|
2089
|
-
|
1908
|
+
disabled: action.disabled,
|
1909
|
+
onSelect: handleClick(action),
|
1910
|
+
display: "block",
|
1911
|
+
children: /* @__PURE__ */ jsxs(Flex, { justifyContent: "space-between", gap: 4, children: [
|
1912
|
+
/* @__PURE__ */ jsxs(
|
1913
|
+
Flex,
|
1914
|
+
{
|
1915
|
+
color: !action.disabled ? convertActionVariantToColor(action.variant) : "inherit",
|
1916
|
+
gap: 2,
|
1917
|
+
tag: "span",
|
1918
|
+
children: [
|
1919
|
+
/* @__PURE__ */ jsx(
|
1920
|
+
Flex,
|
1921
|
+
{
|
1922
|
+
tag: "span",
|
1923
|
+
color: !action.disabled ? convertActionVariantToIconColor(action.variant) : "inherit",
|
1924
|
+
children: action.icon
|
1925
|
+
}
|
1926
|
+
),
|
1927
|
+
action.label
|
1928
|
+
]
|
1929
|
+
}
|
1930
|
+
),
|
1931
|
+
action.id.startsWith("HistoryAction") && /* @__PURE__ */ jsx(
|
1932
|
+
Flex,
|
1933
|
+
{
|
1934
|
+
alignItems: "center",
|
1935
|
+
background: "alternative100",
|
1936
|
+
borderStyle: "solid",
|
1937
|
+
borderColor: "alternative200",
|
1938
|
+
borderWidth: "1px",
|
1939
|
+
height: 5,
|
1940
|
+
paddingLeft: 2,
|
1941
|
+
paddingRight: 2,
|
1942
|
+
hasRadius: true,
|
1943
|
+
color: "alternative600",
|
1944
|
+
children: /* @__PURE__ */ jsx(Typography, { variant: "sigma", fontWeight: "bold", lineHeight: 1, children: formatMessage({ id: "global.new", defaultMessage: "New" }) })
|
1945
|
+
}
|
1946
|
+
)
|
1947
|
+
] })
|
2090
1948
|
},
|
2091
|
-
|
1949
|
+
action.id
|
2092
1950
|
);
|
2093
|
-
}
|
2094
|
-
|
2095
|
-
|
2096
|
-
|
1951
|
+
}),
|
1952
|
+
children
|
1953
|
+
] }),
|
1954
|
+
actions2.map((action) => {
|
1955
|
+
return /* @__PURE__ */ jsxs(React.Fragment, { children: [
|
1956
|
+
action.dialog?.type === "dialog" ? /* @__PURE__ */ jsx(
|
1957
|
+
DocumentActionConfirmDialog,
|
1958
|
+
{
|
1959
|
+
...action.dialog,
|
1960
|
+
variant: action.variant,
|
1961
|
+
isOpen: dialogId === action.id,
|
1962
|
+
onClose: handleClose
|
1963
|
+
}
|
1964
|
+
) : null,
|
1965
|
+
action.dialog?.type === "modal" ? /* @__PURE__ */ jsx(
|
1966
|
+
DocumentActionModal,
|
1967
|
+
{
|
1968
|
+
...action.dialog,
|
1969
|
+
onModalClose: handleClose,
|
1970
|
+
isOpen: dialogId === action.id
|
1971
|
+
}
|
1972
|
+
) : null
|
1973
|
+
] }, action.id);
|
1974
|
+
})
|
1975
|
+
] });
|
1976
|
+
};
|
1977
|
+
const convertActionVariantToColor = (variant = "secondary") => {
|
1978
|
+
switch (variant) {
|
1979
|
+
case "danger":
|
1980
|
+
return "danger600";
|
1981
|
+
case "secondary":
|
1982
|
+
return void 0;
|
1983
|
+
case "success":
|
1984
|
+
return "success600";
|
1985
|
+
default:
|
1986
|
+
return "primary600";
|
1987
|
+
}
|
1988
|
+
};
|
1989
|
+
const convertActionVariantToIconColor = (variant = "secondary") => {
|
1990
|
+
switch (variant) {
|
1991
|
+
case "danger":
|
1992
|
+
return "danger600";
|
1993
|
+
case "secondary":
|
1994
|
+
return "neutral500";
|
1995
|
+
case "success":
|
1996
|
+
return "success600";
|
1997
|
+
default:
|
1998
|
+
return "primary600";
|
1999
|
+
}
|
2000
|
+
};
|
2001
|
+
const DocumentActionConfirmDialog = ({
|
2002
|
+
onClose,
|
2003
|
+
onCancel,
|
2004
|
+
onConfirm,
|
2005
|
+
title,
|
2006
|
+
content,
|
2007
|
+
isOpen,
|
2008
|
+
variant = "secondary"
|
2009
|
+
}) => {
|
2010
|
+
const { formatMessage } = useIntl();
|
2011
|
+
const handleClose = async () => {
|
2012
|
+
if (onCancel) {
|
2013
|
+
await onCancel();
|
2014
|
+
}
|
2015
|
+
onClose();
|
2016
|
+
};
|
2017
|
+
const handleConfirm = async () => {
|
2018
|
+
if (onConfirm) {
|
2019
|
+
await onConfirm();
|
2020
|
+
}
|
2021
|
+
onClose();
|
2097
2022
|
};
|
2023
|
+
return /* @__PURE__ */ jsx(Dialog.Root, { open: isOpen, onOpenChange: handleClose, children: /* @__PURE__ */ jsxs(Dialog.Content, { children: [
|
2024
|
+
/* @__PURE__ */ jsx(Dialog.Header, { children: title }),
|
2025
|
+
/* @__PURE__ */ jsx(Dialog.Body, { children: content }),
|
2026
|
+
/* @__PURE__ */ jsxs(Dialog.Footer, { children: [
|
2027
|
+
/* @__PURE__ */ jsx(Dialog.Cancel, { children: /* @__PURE__ */ jsx(Button, { variant: "tertiary", fullWidth: true, children: formatMessage({
|
2028
|
+
id: "app.components.Button.cancel",
|
2029
|
+
defaultMessage: "Cancel"
|
2030
|
+
}) }) }),
|
2031
|
+
/* @__PURE__ */ jsx(Button, { onClick: handleConfirm, variant, fullWidth: true, children: formatMessage({
|
2032
|
+
id: "app.components.Button.confirm",
|
2033
|
+
defaultMessage: "Confirm"
|
2034
|
+
}) })
|
2035
|
+
] })
|
2036
|
+
] }) });
|
2098
2037
|
};
|
2099
|
-
|
2100
|
-
|
2101
|
-
|
2102
|
-
|
2103
|
-
|
2104
|
-
|
2105
|
-
|
2038
|
+
const DocumentActionModal = ({
|
2039
|
+
isOpen,
|
2040
|
+
title,
|
2041
|
+
onClose,
|
2042
|
+
footer: Footer,
|
2043
|
+
content: Content,
|
2044
|
+
onModalClose
|
2106
2045
|
}) => {
|
2107
|
-
const
|
2108
|
-
|
2109
|
-
|
2110
|
-
const { discard } = useDocumentActions();
|
2111
|
-
const [{ query }] = useQueryParams();
|
2112
|
-
const params = React.useMemo(() => buildValidParams(query), [query]);
|
2113
|
-
if (!schema?.options?.draftAndPublish) {
|
2114
|
-
return null;
|
2115
|
-
}
|
2116
|
-
return {
|
2117
|
-
disabled: !canUpdate || activeTab === "published" || document?.status !== "modified",
|
2118
|
-
label: formatMessage({
|
2119
|
-
id: "content-manager.actions.discard.label",
|
2120
|
-
defaultMessage: "Discard changes"
|
2121
|
-
}),
|
2122
|
-
icon: /* @__PURE__ */ jsx(StyledCrossCircle, {}),
|
2123
|
-
position: ["panel", "table-row"],
|
2124
|
-
variant: "danger",
|
2125
|
-
dialog: {
|
2126
|
-
type: "dialog",
|
2127
|
-
title: formatMessage({
|
2128
|
-
id: "app.components.ConfirmDialog.title",
|
2129
|
-
defaultMessage: "Confirmation"
|
2130
|
-
}),
|
2131
|
-
content: /* @__PURE__ */ jsxs(Flex, { direction: "column", gap: 2, children: [
|
2132
|
-
/* @__PURE__ */ jsx(WarningCircle, { width: "24px", height: "24px", fill: "danger600" }),
|
2133
|
-
/* @__PURE__ */ jsx(Typography, { tag: "p", variant: "omega", textAlign: "center", children: formatMessage({
|
2134
|
-
id: "content-manager.actions.discard.dialog.body",
|
2135
|
-
defaultMessage: "Are you sure?"
|
2136
|
-
}) })
|
2137
|
-
] }),
|
2138
|
-
onConfirm: async () => {
|
2139
|
-
await discard({
|
2140
|
-
collectionType,
|
2141
|
-
model,
|
2142
|
-
documentId,
|
2143
|
-
params
|
2144
|
-
});
|
2145
|
-
}
|
2046
|
+
const handleClose = () => {
|
2047
|
+
if (onClose) {
|
2048
|
+
onClose();
|
2146
2049
|
}
|
2050
|
+
onModalClose();
|
2147
2051
|
};
|
2052
|
+
return /* @__PURE__ */ jsx(Modal.Root, { open: isOpen, onOpenChange: handleClose, children: /* @__PURE__ */ jsxs(Modal.Content, { children: [
|
2053
|
+
/* @__PURE__ */ jsx(Modal.Header, { children: /* @__PURE__ */ jsx(Modal.Title, { children: title }) }),
|
2054
|
+
typeof Content === "function" ? /* @__PURE__ */ jsx(Content, { onClose: handleClose }) : /* @__PURE__ */ jsx(Modal.Body, { children: Content }),
|
2055
|
+
typeof Footer === "function" ? /* @__PURE__ */ jsx(Footer, { onClose: handleClose }) : Footer
|
2056
|
+
] }) });
|
2148
2057
|
};
|
2149
|
-
|
2150
|
-
|
2151
|
-
|
2152
|
-
fill: currentColor;
|
2153
|
-
}
|
2154
|
-
`;
|
2155
|
-
const DEFAULT_ACTIONS = [PublishAction$1, UpdateAction, UnpublishAction$1, DiscardAction];
|
2156
|
-
const intervals = ["years", "months", "days", "hours", "minutes", "seconds"];
|
2157
|
-
const RelativeTime = React.forwardRef(
|
2158
|
-
({ timestamp, customIntervals = [], ...restProps }, forwardedRef) => {
|
2159
|
-
const { formatRelativeTime, formatDate, formatTime } = useIntl();
|
2160
|
-
const interval = intervalToDuration({
|
2161
|
-
start: timestamp,
|
2162
|
-
end: Date.now()
|
2163
|
-
// see https://github.com/date-fns/date-fns/issues/2891 – No idea why it's all partial it returns it every time.
|
2164
|
-
});
|
2165
|
-
const unit = intervals.find((intervalUnit) => {
|
2166
|
-
return interval[intervalUnit] > 0 && Object.keys(interval).includes(intervalUnit);
|
2167
|
-
});
|
2168
|
-
const relativeTime = isPast(timestamp) ? -interval[unit] : interval[unit];
|
2169
|
-
const customInterval = customIntervals.find(
|
2170
|
-
(custom) => interval[custom.unit] < custom.threshold
|
2171
|
-
);
|
2172
|
-
const displayText = customInterval ? customInterval.text : formatRelativeTime(relativeTime, unit, { numeric: "auto" });
|
2173
|
-
return /* @__PURE__ */ jsx(
|
2174
|
-
"time",
|
2175
|
-
{
|
2176
|
-
ref: forwardedRef,
|
2177
|
-
dateTime: timestamp.toISOString(),
|
2178
|
-
role: "time",
|
2179
|
-
title: `${formatDate(timestamp)} ${formatTime(timestamp)}`,
|
2180
|
-
...restProps,
|
2181
|
-
children: displayText
|
2182
|
-
}
|
2183
|
-
);
|
2184
|
-
}
|
2185
|
-
);
|
2186
|
-
const getDisplayName = ({
|
2187
|
-
firstname,
|
2188
|
-
lastname,
|
2189
|
-
username,
|
2190
|
-
email
|
2191
|
-
} = {}) => {
|
2192
|
-
if (username) {
|
2193
|
-
return username;
|
2058
|
+
const transformData = (data) => {
|
2059
|
+
if (Array.isArray(data)) {
|
2060
|
+
return data.map(transformData);
|
2194
2061
|
}
|
2195
|
-
if (
|
2196
|
-
|
2062
|
+
if (typeof data === "object" && data !== null) {
|
2063
|
+
if ("apiData" in data) {
|
2064
|
+
return data.apiData;
|
2065
|
+
}
|
2066
|
+
return mapValues(transformData)(data);
|
2197
2067
|
}
|
2198
|
-
return
|
2199
|
-
};
|
2200
|
-
const capitalise = (str) => str.charAt(0).toUpperCase() + str.slice(1);
|
2201
|
-
const DocumentStatus = ({ status = "draft", ...restProps }) => {
|
2202
|
-
const statusVariant = status === "draft" ? "primary" : status === "published" ? "success" : "alternative";
|
2203
|
-
return /* @__PURE__ */ jsx(Status, { ...restProps, showBullet: false, size: "S", variant: statusVariant, children: /* @__PURE__ */ jsx(Typography, { tag: "span", variant: "omega", fontWeight: "bold", children: capitalise(status) }) });
|
2068
|
+
return data;
|
2204
2069
|
};
|
2205
|
-
const
|
2206
|
-
|
2070
|
+
const PublishAction$1 = ({
|
2071
|
+
activeTab,
|
2072
|
+
documentId,
|
2073
|
+
model,
|
2074
|
+
collectionType,
|
2075
|
+
meta,
|
2076
|
+
document
|
2077
|
+
}) => {
|
2078
|
+
const { schema } = useDoc();
|
2079
|
+
const navigate = useNavigate();
|
2080
|
+
const { toggleNotification } = useNotification();
|
2081
|
+
const { _unstableFormatValidationErrors: formatValidationErrors } = useAPIErrorHandler();
|
2082
|
+
const isListView = useMatch(LIST_PATH) !== null;
|
2207
2083
|
const isCloning = useMatch(CLONE_PATH) !== null;
|
2208
|
-
const title = isCreating ? formatMessage({
|
2209
|
-
id: "content-manager.containers.edit.title.new",
|
2210
|
-
defaultMessage: "Create an entry"
|
2211
|
-
}) : documentTitle;
|
2212
|
-
return /* @__PURE__ */ jsxs(Flex, { direction: "column", alignItems: "flex-start", paddingTop: 6, paddingBottom: 4, gap: 2, children: [
|
2213
|
-
/* @__PURE__ */ jsx(BackButton, {}),
|
2214
|
-
/* @__PURE__ */ jsxs(Flex, { width: "100%", justifyContent: "space-between", gap: "80px", alignItems: "flex-start", children: [
|
2215
|
-
/* @__PURE__ */ jsx(Typography, { variant: "alpha", tag: "h1", children: title }),
|
2216
|
-
/* @__PURE__ */ jsx(HeaderToolbar, {})
|
2217
|
-
] }),
|
2218
|
-
status ? /* @__PURE__ */ jsx(Box, { marginTop: 1, children: /* @__PURE__ */ jsx(DocumentStatus, { status: isCloning ? "draft" : status }) }) : null
|
2219
|
-
] });
|
2220
|
-
};
|
2221
|
-
const HeaderToolbar = () => {
|
2222
2084
|
const { formatMessage } = useIntl();
|
2223
|
-
const
|
2085
|
+
const canPublish = useDocumentRBAC("PublishAction", ({ canPublish: canPublish2 }) => canPublish2);
|
2086
|
+
const { publish } = useDocumentActions();
|
2224
2087
|
const [
|
2225
|
-
|
2226
|
-
|
2088
|
+
countDraftRelations,
|
2089
|
+
{ isLoading: isLoadingDraftRelations, isError: isErrorDraftRelations }
|
2090
|
+
] = useLazyGetDraftRelationCountQuery();
|
2091
|
+
const [localCountOfDraftRelations, setLocalCountOfDraftRelations] = React.useState(0);
|
2092
|
+
const [serverCountOfDraftRelations, setServerCountOfDraftRelations] = React.useState(0);
|
2093
|
+
const [{ query, rawQuery }] = useQueryParams();
|
2094
|
+
const params = React.useMemo(() => buildValidParams(query), [query]);
|
2095
|
+
const modified = useForm("PublishAction", ({ modified: modified2 }) => modified2);
|
2096
|
+
const setSubmitting = useForm("PublishAction", ({ setSubmitting: setSubmitting2 }) => setSubmitting2);
|
2097
|
+
const isSubmitting = useForm("PublishAction", ({ isSubmitting: isSubmitting2 }) => isSubmitting2);
|
2098
|
+
const validate = useForm("PublishAction", (state) => state.validate);
|
2099
|
+
const setErrors = useForm("PublishAction", (state) => state.setErrors);
|
2100
|
+
const formValues = useForm("PublishAction", ({ values }) => values);
|
2101
|
+
React.useEffect(() => {
|
2102
|
+
if (isErrorDraftRelations) {
|
2103
|
+
toggleNotification({
|
2104
|
+
type: "danger",
|
2105
|
+
message: formatMessage({
|
2106
|
+
id: getTranslation("error.records.fetch-draft-relatons"),
|
2107
|
+
defaultMessage: "An error occurred while fetching draft relations on this document."
|
2108
|
+
})
|
2109
|
+
});
|
2227
2110
|
}
|
2228
|
-
|
2229
|
-
|
2230
|
-
|
2231
|
-
|
2232
|
-
|
2233
|
-
|
2234
|
-
|
2235
|
-
|
2236
|
-
activeTab: status,
|
2237
|
-
model,
|
2238
|
-
documentId: id,
|
2239
|
-
document: isCloning ? void 0 : document,
|
2240
|
-
meta: isCloning ? void 0 : meta,
|
2241
|
-
collectionType
|
2242
|
-
},
|
2243
|
-
descriptions: plugins["content-manager"].apis.getHeaderActions(),
|
2244
|
-
children: (actions2) => {
|
2245
|
-
if (actions2.length > 0) {
|
2246
|
-
return /* @__PURE__ */ jsx(HeaderActions, { actions: actions2 });
|
2247
|
-
} else {
|
2248
|
-
return null;
|
2249
|
-
}
|
2111
|
+
}, [isErrorDraftRelations, toggleNotification, formatMessage]);
|
2112
|
+
React.useEffect(() => {
|
2113
|
+
const localDraftRelations = /* @__PURE__ */ new Set();
|
2114
|
+
const extractDraftRelations = (data) => {
|
2115
|
+
const relations = data.connect || [];
|
2116
|
+
relations.forEach((relation) => {
|
2117
|
+
if (relation.status === "draft") {
|
2118
|
+
localDraftRelations.add(relation.id);
|
2250
2119
|
}
|
2251
|
-
}
|
2252
|
-
|
2253
|
-
|
2254
|
-
|
2255
|
-
|
2256
|
-
|
2257
|
-
|
2258
|
-
|
2259
|
-
documentId: id,
|
2260
|
-
document: isCloning ? void 0 : document,
|
2261
|
-
meta: isCloning ? void 0 : meta,
|
2262
|
-
collectionType
|
2263
|
-
},
|
2264
|
-
descriptions: plugins["content-manager"].apis.getDocumentActions(),
|
2265
|
-
children: (actions2) => {
|
2266
|
-
const headerActions = actions2.filter((action) => {
|
2267
|
-
const positions = Array.isArray(action.position) ? action.position : [action.position];
|
2268
|
-
return positions.includes("header");
|
2269
|
-
});
|
2270
|
-
return /* @__PURE__ */ jsx(
|
2271
|
-
DocumentActionsMenu,
|
2272
|
-
{
|
2273
|
-
actions: headerActions,
|
2274
|
-
label: formatMessage({
|
2275
|
-
id: "content-manager.containers.edit.header.more-actions",
|
2276
|
-
defaultMessage: "More actions"
|
2277
|
-
}),
|
2278
|
-
children: /* @__PURE__ */ jsx(Information, { activeTab: status })
|
2279
|
-
}
|
2280
|
-
);
|
2120
|
+
});
|
2121
|
+
};
|
2122
|
+
const traverseAndExtract = (data) => {
|
2123
|
+
Object.entries(data).forEach(([key, value]) => {
|
2124
|
+
if (key === "connect" && Array.isArray(value)) {
|
2125
|
+
extractDraftRelations({ connect: value });
|
2126
|
+
} else if (typeof value === "object" && value !== null) {
|
2127
|
+
traverseAndExtract(value);
|
2281
2128
|
}
|
2129
|
+
});
|
2130
|
+
};
|
2131
|
+
if (!documentId || modified) {
|
2132
|
+
traverseAndExtract(formValues);
|
2133
|
+
setLocalCountOfDraftRelations(localDraftRelations.size);
|
2134
|
+
}
|
2135
|
+
}, [documentId, modified, formValues, setLocalCountOfDraftRelations]);
|
2136
|
+
React.useEffect(() => {
|
2137
|
+
if (!document || !document.documentId || isListView) {
|
2138
|
+
return;
|
2139
|
+
}
|
2140
|
+
const fetchDraftRelationsCount = async () => {
|
2141
|
+
const { data, error } = await countDraftRelations({
|
2142
|
+
collectionType,
|
2143
|
+
model,
|
2144
|
+
documentId,
|
2145
|
+
params
|
2146
|
+
});
|
2147
|
+
if (error) {
|
2148
|
+
throw error;
|
2282
2149
|
}
|
2283
|
-
|
2284
|
-
|
2285
|
-
}
|
2286
|
-
|
2287
|
-
|
2288
|
-
|
2289
|
-
|
2150
|
+
if (data) {
|
2151
|
+
setServerCountOfDraftRelations(data.data);
|
2152
|
+
}
|
2153
|
+
};
|
2154
|
+
fetchDraftRelationsCount();
|
2155
|
+
}, [isListView, document, documentId, countDraftRelations, collectionType, model, params]);
|
2156
|
+
const isDocumentPublished = (document?.[PUBLISHED_AT_ATTRIBUTE_NAME] || meta?.availableStatus.some((doc) => doc[PUBLISHED_AT_ATTRIBUTE_NAME] !== null)) && document?.status !== "modified";
|
2157
|
+
if (!schema?.options?.draftAndPublish) {
|
2290
2158
|
return null;
|
2291
2159
|
}
|
2292
|
-
const
|
2293
|
-
|
2294
|
-
|
2295
|
-
|
2296
|
-
|
2297
|
-
|
2298
|
-
|
2299
|
-
|
2300
|
-
|
2301
|
-
|
2302
|
-
|
2303
|
-
|
2160
|
+
const performPublish = async () => {
|
2161
|
+
setSubmitting(true);
|
2162
|
+
try {
|
2163
|
+
const { errors } = await validate(true, {
|
2164
|
+
status: "published"
|
2165
|
+
});
|
2166
|
+
if (errors) {
|
2167
|
+
toggleNotification({
|
2168
|
+
type: "danger",
|
2169
|
+
message: formatMessage({
|
2170
|
+
id: "content-manager.validation.error",
|
2171
|
+
defaultMessage: "There are validation errors in your document. Please fix them before saving."
|
2172
|
+
})
|
2173
|
+
});
|
2174
|
+
return;
|
2175
|
+
}
|
2176
|
+
const res = await publish(
|
2304
2177
|
{
|
2305
|
-
|
2306
|
-
|
2178
|
+
collectionType,
|
2179
|
+
model,
|
2180
|
+
documentId,
|
2181
|
+
params
|
2307
2182
|
},
|
2308
|
-
|
2309
|
-
|
2310
|
-
|
2311
|
-
|
2312
|
-
|
2313
|
-
|
2183
|
+
transformData(formValues)
|
2184
|
+
);
|
2185
|
+
if ("data" in res && collectionType !== SINGLE_TYPES) {
|
2186
|
+
navigate({
|
2187
|
+
pathname: `../${collectionType}/${model}/${res.data.documentId}`,
|
2188
|
+
search: rawQuery
|
2189
|
+
});
|
2190
|
+
} else if ("error" in res && isBaseQueryError(res.error) && res.error.name === "ValidationError") {
|
2191
|
+
setErrors(formatValidationErrors(res.error));
|
2192
|
+
}
|
2193
|
+
} finally {
|
2194
|
+
setSubmitting(false);
|
2195
|
+
}
|
2196
|
+
};
|
2197
|
+
const totalDraftRelations = localCountOfDraftRelations + serverCountOfDraftRelations;
|
2198
|
+
const enableDraftRelationsCount = false;
|
2199
|
+
const hasDraftRelations = enableDraftRelationsCount;
|
2200
|
+
return {
|
2201
|
+
/**
|
2202
|
+
* Disabled when:
|
2203
|
+
* - currently if you're cloning a document we don't support publish & clone at the same time.
|
2204
|
+
* - the form is submitting
|
2205
|
+
* - the active tab is the published tab
|
2206
|
+
* - the document is already published & not modified
|
2207
|
+
* - the document is being created & not modified
|
2208
|
+
* - the user doesn't have the permission to publish
|
2209
|
+
*/
|
2210
|
+
disabled: isCloning || isSubmitting || isLoadingDraftRelations || activeTab === "published" || !modified && isDocumentPublished || !modified && !document?.documentId || !canPublish,
|
2211
|
+
label: formatMessage({
|
2212
|
+
id: "app.utils.publish",
|
2213
|
+
defaultMessage: "Publish"
|
2214
|
+
}),
|
2215
|
+
onClick: async () => {
|
2216
|
+
await performPublish();
|
2314
2217
|
},
|
2315
|
-
{
|
2316
|
-
|
2317
|
-
|
2318
|
-
|
2319
|
-
|
2218
|
+
dialog: hasDraftRelations ? {
|
2219
|
+
type: "dialog",
|
2220
|
+
variant: "danger",
|
2221
|
+
footer: null,
|
2222
|
+
title: formatMessage({
|
2223
|
+
id: getTranslation(`popUpwarning.warning.bulk-has-draft-relations.title`),
|
2224
|
+
defaultMessage: "Confirmation"
|
2320
2225
|
}),
|
2321
|
-
|
2226
|
+
content: formatMessage(
|
2322
2227
|
{
|
2323
|
-
id:
|
2324
|
-
defaultMessage:
|
2228
|
+
id: getTranslation(`popUpwarning.warning.bulk-has-draft-relations.message`),
|
2229
|
+
defaultMessage: "This entry is related to {count, plural, one {# draft entry} other {# draft entries}}. Publishing it could leave broken links in your app."
|
2325
2230
|
},
|
2326
2231
|
{
|
2327
|
-
|
2328
|
-
RelativeTime,
|
2329
|
-
{
|
2330
|
-
timestamp: new Date(createAndUpdateDocument?.[UPDATED_AT_ATTRIBUTE_NAME])
|
2331
|
-
}
|
2332
|
-
),
|
2333
|
-
isAnonymous: !updator,
|
2334
|
-
author: updator
|
2232
|
+
count: totalDraftRelations
|
2335
2233
|
}
|
2336
|
-
)
|
2337
|
-
|
2338
|
-
|
2339
|
-
|
2340
|
-
|
2341
|
-
|
2342
|
-
|
2343
|
-
|
2344
|
-
|
2345
|
-
|
2346
|
-
|
2347
|
-
|
2348
|
-
|
2349
|
-
|
2350
|
-
|
2351
|
-
|
2352
|
-
|
2353
|
-
|
2354
|
-
|
2355
|
-
|
2356
|
-
|
2357
|
-
|
2234
|
+
),
|
2235
|
+
onConfirm: async () => {
|
2236
|
+
await performPublish();
|
2237
|
+
}
|
2238
|
+
} : void 0
|
2239
|
+
};
|
2240
|
+
};
|
2241
|
+
PublishAction$1.type = "publish";
|
2242
|
+
const UpdateAction = ({
|
2243
|
+
activeTab,
|
2244
|
+
documentId,
|
2245
|
+
model,
|
2246
|
+
collectionType
|
2247
|
+
}) => {
|
2248
|
+
const navigate = useNavigate();
|
2249
|
+
const { toggleNotification } = useNotification();
|
2250
|
+
const { _unstableFormatValidationErrors: formatValidationErrors } = useAPIErrorHandler();
|
2251
|
+
const cloneMatch = useMatch(CLONE_PATH);
|
2252
|
+
const isCloning = cloneMatch !== null;
|
2253
|
+
const { formatMessage } = useIntl();
|
2254
|
+
const { create, update, clone } = useDocumentActions();
|
2255
|
+
const [{ query, rawQuery }] = useQueryParams();
|
2256
|
+
const params = React.useMemo(() => buildValidParams(query), [query]);
|
2257
|
+
const isSubmitting = useForm("UpdateAction", ({ isSubmitting: isSubmitting2 }) => isSubmitting2);
|
2258
|
+
const modified = useForm("UpdateAction", ({ modified: modified2 }) => modified2);
|
2259
|
+
const setSubmitting = useForm("UpdateAction", ({ setSubmitting: setSubmitting2 }) => setSubmitting2);
|
2260
|
+
const document = useForm("UpdateAction", ({ values }) => values);
|
2261
|
+
const validate = useForm("UpdateAction", (state) => state.validate);
|
2262
|
+
const setErrors = useForm("UpdateAction", (state) => state.setErrors);
|
2263
|
+
const resetForm = useForm("PublishAction", ({ resetForm: resetForm2 }) => resetForm2);
|
2264
|
+
return {
|
2265
|
+
/**
|
2266
|
+
* Disabled when:
|
2267
|
+
* - the form is submitting
|
2268
|
+
* - the document is not modified & we're not cloning (you can save a clone entity straight away)
|
2269
|
+
* - the active tab is the published tab
|
2270
|
+
*/
|
2271
|
+
disabled: isSubmitting || !modified && !isCloning || activeTab === "published",
|
2272
|
+
label: formatMessage({
|
2273
|
+
id: "content-manager.containers.Edit.save",
|
2274
|
+
defaultMessage: "Save"
|
2275
|
+
}),
|
2276
|
+
onClick: async () => {
|
2277
|
+
setSubmitting(true);
|
2278
|
+
try {
|
2279
|
+
const { errors } = await validate(true, {
|
2280
|
+
status: "draft"
|
2281
|
+
});
|
2282
|
+
if (errors) {
|
2283
|
+
toggleNotification({
|
2284
|
+
type: "danger",
|
2285
|
+
message: formatMessage({
|
2286
|
+
id: "content-manager.validation.error",
|
2287
|
+
defaultMessage: "There are validation errors in your document. Please fix them before saving."
|
2288
|
+
})
|
2289
|
+
});
|
2290
|
+
return;
|
2358
2291
|
}
|
2359
|
-
|
2360
|
-
|
2361
|
-
|
2362
|
-
|
2363
|
-
|
2364
|
-
|
2365
|
-
|
2366
|
-
|
2367
|
-
|
2368
|
-
|
2369
|
-
|
2370
|
-
|
2371
|
-
|
2372
|
-
|
2373
|
-
|
2374
|
-
|
2375
|
-
|
2376
|
-
|
2377
|
-
|
2378
|
-
|
2379
|
-
|
2380
|
-
|
2292
|
+
if (isCloning) {
|
2293
|
+
const res = await clone(
|
2294
|
+
{
|
2295
|
+
model,
|
2296
|
+
documentId: cloneMatch.params.origin,
|
2297
|
+
params
|
2298
|
+
},
|
2299
|
+
transformData(document)
|
2300
|
+
);
|
2301
|
+
if ("data" in res) {
|
2302
|
+
navigate(
|
2303
|
+
{
|
2304
|
+
pathname: `../${res.data.documentId}`,
|
2305
|
+
search: rawQuery
|
2306
|
+
},
|
2307
|
+
{ relative: "path" }
|
2308
|
+
);
|
2309
|
+
} else if ("error" in res && isBaseQueryError(res.error) && res.error.name === "ValidationError") {
|
2310
|
+
setErrors(formatValidationErrors(res.error));
|
2311
|
+
}
|
2312
|
+
} else if (documentId || collectionType === SINGLE_TYPES) {
|
2313
|
+
const res = await update(
|
2314
|
+
{
|
2315
|
+
collectionType,
|
2316
|
+
model,
|
2317
|
+
documentId,
|
2318
|
+
params
|
2319
|
+
},
|
2320
|
+
transformData(document)
|
2321
|
+
);
|
2322
|
+
if ("error" in res && isBaseQueryError(res.error) && res.error.name === "ValidationError") {
|
2323
|
+
setErrors(formatValidationErrors(res.error));
|
2324
|
+
} else {
|
2325
|
+
resetForm();
|
2326
|
+
}
|
2327
|
+
} else {
|
2328
|
+
const res = await create(
|
2329
|
+
{
|
2330
|
+
model,
|
2331
|
+
params
|
2332
|
+
},
|
2333
|
+
transformData(document)
|
2334
|
+
);
|
2335
|
+
if ("data" in res && collectionType !== SINGLE_TYPES) {
|
2336
|
+
navigate(
|
2337
|
+
{
|
2338
|
+
pathname: `../${res.data.documentId}`,
|
2339
|
+
search: rawQuery
|
2340
|
+
},
|
2341
|
+
{ replace: true, relative: "path" }
|
2342
|
+
);
|
2343
|
+
} else if ("error" in res && isBaseQueryError(res.error) && res.error.name === "ValidationError") {
|
2344
|
+
setErrors(formatValidationErrors(res.error));
|
2345
|
+
}
|
2346
|
+
}
|
2347
|
+
} finally {
|
2348
|
+
setSubmitting(false);
|
2349
|
+
}
|
2381
2350
|
}
|
2382
|
-
|
2351
|
+
};
|
2383
2352
|
};
|
2384
|
-
|
2385
|
-
|
2386
|
-
|
2387
|
-
|
2388
|
-
SingleSelect,
|
2389
|
-
{
|
2390
|
-
size: "S",
|
2391
|
-
disabled: action.disabled,
|
2392
|
-
"aria-label": action.label,
|
2393
|
-
onChange: action.onSelect,
|
2394
|
-
value: action.value,
|
2395
|
-
children: action.options.map(({ label, ...option }) => /* @__PURE__ */ jsx(SingleSelectOption, { ...option, children: label }, option.value))
|
2396
|
-
},
|
2397
|
-
action.id
|
2398
|
-
);
|
2399
|
-
} else {
|
2400
|
-
return null;
|
2401
|
-
}
|
2402
|
-
}) });
|
2353
|
+
UpdateAction.type = "update";
|
2354
|
+
const UNPUBLISH_DRAFT_OPTIONS = {
|
2355
|
+
KEEP: "keep",
|
2356
|
+
DISCARD: "discard"
|
2403
2357
|
};
|
2404
|
-
const
|
2405
|
-
|
2358
|
+
const UnpublishAction$1 = ({
|
2359
|
+
activeTab,
|
2360
|
+
documentId,
|
2361
|
+
model,
|
2362
|
+
collectionType,
|
2363
|
+
document
|
2364
|
+
}) => {
|
2406
2365
|
const { formatMessage } = useIntl();
|
2407
|
-
|
2408
|
-
|
2409
|
-
|
2410
|
-
|
2411
|
-
|
2412
|
-
|
2413
|
-
|
2414
|
-
|
2415
|
-
|
2416
|
-
|
2366
|
+
const { schema } = useDoc();
|
2367
|
+
const canPublish = useDocumentRBAC("UnpublishAction", ({ canPublish: canPublish2 }) => canPublish2);
|
2368
|
+
const { unpublish } = useDocumentActions();
|
2369
|
+
const [{ query }] = useQueryParams();
|
2370
|
+
const params = React.useMemo(() => buildValidParams(query), [query]);
|
2371
|
+
const { toggleNotification } = useNotification();
|
2372
|
+
const [shouldKeepDraft, setShouldKeepDraft] = React.useState(true);
|
2373
|
+
const isDocumentModified = document?.status === "modified";
|
2374
|
+
const handleChange = (value) => {
|
2375
|
+
setShouldKeepDraft(value === UNPUBLISH_DRAFT_OPTIONS.KEEP);
|
2417
2376
|
};
|
2418
|
-
|
2419
|
-
|
2420
|
-
|
2421
|
-
const navigate = useNavigate();
|
2422
|
-
const { formatMessage } = useIntl();
|
2377
|
+
if (!schema?.options?.draftAndPublish) {
|
2378
|
+
return null;
|
2379
|
+
}
|
2423
2380
|
return {
|
2381
|
+
disabled: !canPublish || activeTab === "published" || document?.status !== "published" && document?.status !== "modified",
|
2424
2382
|
label: formatMessage({
|
2425
|
-
id: "
|
2426
|
-
defaultMessage: "
|
2383
|
+
id: "app.utils.unpublish",
|
2384
|
+
defaultMessage: "Unpublish"
|
2427
2385
|
}),
|
2428
|
-
icon: /* @__PURE__ */ jsx(
|
2429
|
-
onClick: () => {
|
2430
|
-
|
2386
|
+
icon: /* @__PURE__ */ jsx(Cross, {}),
|
2387
|
+
onClick: async () => {
|
2388
|
+
if (!documentId && collectionType !== SINGLE_TYPES || isDocumentModified) {
|
2389
|
+
if (!documentId) {
|
2390
|
+
console.error(
|
2391
|
+
"You're trying to unpublish a document without an id, this is likely a bug with Strapi. Please open an issue."
|
2392
|
+
);
|
2393
|
+
toggleNotification({
|
2394
|
+
message: formatMessage({
|
2395
|
+
id: "content-manager.actions.unpublish.error",
|
2396
|
+
defaultMessage: "An error occurred while trying to unpublish the document."
|
2397
|
+
}),
|
2398
|
+
type: "danger"
|
2399
|
+
});
|
2400
|
+
}
|
2401
|
+
return;
|
2402
|
+
}
|
2403
|
+
await unpublish({
|
2404
|
+
collectionType,
|
2405
|
+
model,
|
2406
|
+
documentId,
|
2407
|
+
params
|
2408
|
+
});
|
2431
2409
|
},
|
2432
|
-
|
2410
|
+
dialog: isDocumentModified ? {
|
2411
|
+
type: "dialog",
|
2412
|
+
title: formatMessage({
|
2413
|
+
id: "app.components.ConfirmDialog.title",
|
2414
|
+
defaultMessage: "Confirmation"
|
2415
|
+
}),
|
2416
|
+
content: /* @__PURE__ */ jsxs(Flex, { alignItems: "flex-start", direction: "column", gap: 6, children: [
|
2417
|
+
/* @__PURE__ */ jsxs(Flex, { width: "100%", direction: "column", gap: 2, children: [
|
2418
|
+
/* @__PURE__ */ jsx(WarningCircle, { width: "24px", height: "24px", fill: "danger600" }),
|
2419
|
+
/* @__PURE__ */ jsx(Typography, { tag: "p", variant: "omega", textAlign: "center", children: formatMessage({
|
2420
|
+
id: "content-manager.actions.unpublish.dialog.body",
|
2421
|
+
defaultMessage: "Are you sure?"
|
2422
|
+
}) })
|
2423
|
+
] }),
|
2424
|
+
/* @__PURE__ */ jsxs(
|
2425
|
+
Radio.Group,
|
2426
|
+
{
|
2427
|
+
defaultValue: UNPUBLISH_DRAFT_OPTIONS.KEEP,
|
2428
|
+
name: "discard-options",
|
2429
|
+
"aria-label": formatMessage({
|
2430
|
+
id: "content-manager.actions.unpublish.dialog.radio-label",
|
2431
|
+
defaultMessage: "Choose an option to unpublish the document."
|
2432
|
+
}),
|
2433
|
+
onValueChange: handleChange,
|
2434
|
+
children: [
|
2435
|
+
/* @__PURE__ */ jsx(Radio.Item, { checked: shouldKeepDraft, value: UNPUBLISH_DRAFT_OPTIONS.KEEP, children: formatMessage({
|
2436
|
+
id: "content-manager.actions.unpublish.dialog.option.keep-draft",
|
2437
|
+
defaultMessage: "Keep draft"
|
2438
|
+
}) }),
|
2439
|
+
/* @__PURE__ */ jsx(Radio.Item, { checked: !shouldKeepDraft, value: UNPUBLISH_DRAFT_OPTIONS.DISCARD, children: formatMessage({
|
2440
|
+
id: "content-manager.actions.unpublish.dialog.option.replace-draft",
|
2441
|
+
defaultMessage: "Replace draft"
|
2442
|
+
}) })
|
2443
|
+
]
|
2444
|
+
}
|
2445
|
+
)
|
2446
|
+
] }),
|
2447
|
+
onConfirm: async () => {
|
2448
|
+
if (!documentId && collectionType !== SINGLE_TYPES) {
|
2449
|
+
console.error(
|
2450
|
+
"You're trying to unpublish a document without an id, this is likely a bug with Strapi. Please open an issue."
|
2451
|
+
);
|
2452
|
+
toggleNotification({
|
2453
|
+
message: formatMessage({
|
2454
|
+
id: "content-manager.actions.unpublish.error",
|
2455
|
+
defaultMessage: "An error occurred while trying to unpublish the document."
|
2456
|
+
}),
|
2457
|
+
type: "danger"
|
2458
|
+
});
|
2459
|
+
}
|
2460
|
+
await unpublish(
|
2461
|
+
{
|
2462
|
+
collectionType,
|
2463
|
+
model,
|
2464
|
+
documentId,
|
2465
|
+
params
|
2466
|
+
},
|
2467
|
+
!shouldKeepDraft
|
2468
|
+
);
|
2469
|
+
}
|
2470
|
+
} : void 0,
|
2471
|
+
variant: "danger",
|
2472
|
+
position: ["panel", "table-row"]
|
2433
2473
|
};
|
2434
2474
|
};
|
2435
|
-
|
2436
|
-
const
|
2437
|
-
|
2475
|
+
UnpublishAction$1.type = "unpublish";
|
2476
|
+
const DiscardAction = ({
|
2477
|
+
activeTab,
|
2478
|
+
documentId,
|
2479
|
+
model,
|
2480
|
+
collectionType,
|
2481
|
+
document
|
2482
|
+
}) => {
|
2438
2483
|
const { formatMessage } = useIntl();
|
2439
|
-
const
|
2440
|
-
const
|
2441
|
-
const {
|
2442
|
-
const {
|
2443
|
-
const
|
2484
|
+
const { schema } = useDoc();
|
2485
|
+
const canUpdate = useDocumentRBAC("DiscardAction", ({ canUpdate: canUpdate2 }) => canUpdate2);
|
2486
|
+
const { discard } = useDocumentActions();
|
2487
|
+
const [{ query }] = useQueryParams();
|
2488
|
+
const params = React.useMemo(() => buildValidParams(query), [query]);
|
2489
|
+
if (!schema?.options?.draftAndPublish) {
|
2490
|
+
return null;
|
2491
|
+
}
|
2444
2492
|
return {
|
2445
|
-
disabled: !
|
2493
|
+
disabled: !canUpdate || activeTab === "published" || document?.status !== "modified",
|
2446
2494
|
label: formatMessage({
|
2447
|
-
id: "content-manager.actions.
|
2448
|
-
defaultMessage: "
|
2495
|
+
id: "content-manager.actions.discard.label",
|
2496
|
+
defaultMessage: "Discard changes"
|
2449
2497
|
}),
|
2450
|
-
icon: /* @__PURE__ */ jsx(
|
2498
|
+
icon: /* @__PURE__ */ jsx(Cross, {}),
|
2499
|
+
position: ["panel", "table-row"],
|
2500
|
+
variant: "danger",
|
2451
2501
|
dialog: {
|
2452
2502
|
type: "dialog",
|
2453
2503
|
title: formatMessage({
|
@@ -2457,92 +2507,90 @@ const DeleteAction$1 = ({ documentId, model, collectionType, document }) => {
|
|
2457
2507
|
content: /* @__PURE__ */ jsxs(Flex, { direction: "column", gap: 2, children: [
|
2458
2508
|
/* @__PURE__ */ jsx(WarningCircle, { width: "24px", height: "24px", fill: "danger600" }),
|
2459
2509
|
/* @__PURE__ */ jsx(Typography, { tag: "p", variant: "omega", textAlign: "center", children: formatMessage({
|
2460
|
-
id: "content-manager.actions.
|
2510
|
+
id: "content-manager.actions.discard.dialog.body",
|
2461
2511
|
defaultMessage: "Are you sure?"
|
2462
2512
|
}) })
|
2463
2513
|
] }),
|
2464
2514
|
onConfirm: async () => {
|
2465
|
-
|
2466
|
-
|
2467
|
-
|
2468
|
-
|
2469
|
-
|
2470
|
-
|
2471
|
-
|
2472
|
-
|
2473
|
-
|
2474
|
-
|
2475
|
-
|
2476
|
-
|
2477
|
-
|
2478
|
-
|
2479
|
-
|
2480
|
-
|
2481
|
-
|
2482
|
-
|
2483
|
-
|
2484
|
-
|
2485
|
-
|
2486
|
-
|
2487
|
-
|
2488
|
-
|
2489
|
-
|
2490
|
-
|
2491
|
-
|
2492
|
-
|
2493
|
-
|
2494
|
-
|
2495
|
-
|
2496
|
-
|
2497
|
-
|
2515
|
+
await discard({
|
2516
|
+
collectionType,
|
2517
|
+
model,
|
2518
|
+
documentId,
|
2519
|
+
params
|
2520
|
+
});
|
2521
|
+
}
|
2522
|
+
}
|
2523
|
+
};
|
2524
|
+
};
|
2525
|
+
DiscardAction.type = "discard";
|
2526
|
+
const DEFAULT_ACTIONS = [PublishAction$1, UpdateAction, UnpublishAction$1, DiscardAction];
|
2527
|
+
const intervals = ["years", "months", "days", "hours", "minutes", "seconds"];
|
2528
|
+
const RelativeTime = React.forwardRef(
|
2529
|
+
({ timestamp, customIntervals = [], ...restProps }, forwardedRef) => {
|
2530
|
+
const { formatRelativeTime, formatDate, formatTime } = useIntl();
|
2531
|
+
const interval = intervalToDuration({
|
2532
|
+
start: timestamp,
|
2533
|
+
end: Date.now()
|
2534
|
+
// see https://github.com/date-fns/date-fns/issues/2891 – No idea why it's all partial it returns it every time.
|
2535
|
+
});
|
2536
|
+
const unit = intervals.find((intervalUnit) => {
|
2537
|
+
return interval[intervalUnit] > 0 && Object.keys(interval).includes(intervalUnit);
|
2538
|
+
});
|
2539
|
+
const relativeTime = isPast(timestamp) ? -interval[unit] : interval[unit];
|
2540
|
+
const customInterval = customIntervals.find(
|
2541
|
+
(custom) => interval[custom.unit] < custom.threshold
|
2542
|
+
);
|
2543
|
+
const displayText = customInterval ? customInterval.text : formatRelativeTime(relativeTime, unit, { numeric: "auto" });
|
2544
|
+
return /* @__PURE__ */ jsx(
|
2545
|
+
"time",
|
2546
|
+
{
|
2547
|
+
ref: forwardedRef,
|
2548
|
+
dateTime: timestamp.toISOString(),
|
2549
|
+
role: "time",
|
2550
|
+
title: `${formatDate(timestamp)} ${formatTime(timestamp)}`,
|
2551
|
+
...restProps,
|
2552
|
+
children: displayText
|
2498
2553
|
}
|
2499
|
-
|
2500
|
-
|
2501
|
-
|
2502
|
-
|
2554
|
+
);
|
2555
|
+
}
|
2556
|
+
);
|
2557
|
+
const getDisplayName = ({
|
2558
|
+
firstname,
|
2559
|
+
lastname,
|
2560
|
+
username,
|
2561
|
+
email
|
2562
|
+
} = {}) => {
|
2563
|
+
if (username) {
|
2564
|
+
return username;
|
2565
|
+
}
|
2566
|
+
if (firstname) {
|
2567
|
+
return `${firstname} ${lastname ?? ""}`.trim();
|
2568
|
+
}
|
2569
|
+
return email ?? "";
|
2503
2570
|
};
|
2504
|
-
|
2505
|
-
const
|
2506
|
-
const
|
2507
|
-
|
2508
|
-
const [
|
2509
|
-
{
|
2510
|
-
query: { status }
|
2511
|
-
}
|
2512
|
-
] = useQueryParams({
|
2513
|
-
status: "draft"
|
2514
|
-
});
|
2515
|
-
const { model, id, document, meta, collectionType } = useDoc();
|
2516
|
-
const plugins = useStrapiApp("Panels", (state) => state.plugins);
|
2517
|
-
const props = {
|
2518
|
-
activeTab: status,
|
2519
|
-
model,
|
2520
|
-
documentId: id,
|
2521
|
-
document: isCloning ? void 0 : document,
|
2522
|
-
meta: isCloning ? void 0 : meta,
|
2523
|
-
collectionType
|
2524
|
-
};
|
2525
|
-
return /* @__PURE__ */ jsx(Flex, { direction: "column", alignItems: "stretch", gap: 2, children: /* @__PURE__ */ jsx(
|
2526
|
-
DescriptionComponentRenderer,
|
2527
|
-
{
|
2528
|
-
props,
|
2529
|
-
descriptions: plugins["content-manager"].apis.getEditViewSidePanels(),
|
2530
|
-
children: (panels) => panels.map(({ content, id: id2, ...description }) => /* @__PURE__ */ jsx(Panel, { ...description, children: content }, id2))
|
2531
|
-
}
|
2532
|
-
) });
|
2571
|
+
const capitalise = (str) => str.charAt(0).toUpperCase() + str.slice(1);
|
2572
|
+
const DocumentStatus = ({ status = "draft", ...restProps }) => {
|
2573
|
+
const statusVariant = status === "draft" ? "secondary" : status === "published" ? "success" : "alternative";
|
2574
|
+
return /* @__PURE__ */ jsx(Status, { ...restProps, showBullet: false, size: "S", variant: statusVariant, children: /* @__PURE__ */ jsx(Typography, { tag: "span", variant: "omega", fontWeight: "bold", children: capitalise(status) }) });
|
2533
2575
|
};
|
2534
|
-
const
|
2576
|
+
const Header = ({ isCreating, status, title: documentTitle = "Untitled" }) => {
|
2535
2577
|
const { formatMessage } = useIntl();
|
2536
|
-
|
2537
|
-
|
2538
|
-
|
2539
|
-
|
2540
|
-
|
2541
|
-
|
2542
|
-
|
2578
|
+
const isCloning = useMatch(CLONE_PATH) !== null;
|
2579
|
+
const title = isCreating ? formatMessage({
|
2580
|
+
id: "content-manager.containers.edit.title.new",
|
2581
|
+
defaultMessage: "Create an entry"
|
2582
|
+
}) : documentTitle;
|
2583
|
+
return /* @__PURE__ */ jsxs(Flex, { direction: "column", alignItems: "flex-start", paddingTop: 6, paddingBottom: 4, gap: 2, children: [
|
2584
|
+
/* @__PURE__ */ jsx(BackButton, {}),
|
2585
|
+
/* @__PURE__ */ jsxs(Flex, { width: "100%", justifyContent: "space-between", gap: "80px", alignItems: "flex-start", children: [
|
2586
|
+
/* @__PURE__ */ jsx(Typography, { variant: "alpha", tag: "h1", children: title }),
|
2587
|
+
/* @__PURE__ */ jsx(HeaderToolbar, {})
|
2588
|
+
] }),
|
2589
|
+
status ? /* @__PURE__ */ jsx(Box, { marginTop: 1, children: /* @__PURE__ */ jsx(DocumentStatus, { status: isCloning ? "draft" : status }) }) : null
|
2590
|
+
] });
|
2543
2591
|
};
|
2544
|
-
|
2545
|
-
const
|
2592
|
+
const HeaderToolbar = () => {
|
2593
|
+
const { formatMessage } = useIntl();
|
2546
2594
|
const isCloning = useMatch(CLONE_PATH) !== null;
|
2547
2595
|
const [
|
2548
2596
|
{
|
@@ -2550,355 +2598,432 @@ const ActionsPanelContent = () => {
|
|
2550
2598
|
}
|
2551
2599
|
] = useQueryParams();
|
2552
2600
|
const { model, id, document, meta, collectionType } = useDoc();
|
2553
|
-
const plugins = useStrapiApp("
|
2554
|
-
|
2555
|
-
activeTab: status,
|
2556
|
-
model,
|
2557
|
-
documentId: id,
|
2558
|
-
document: isCloning ? void 0 : document,
|
2559
|
-
meta: isCloning ? void 0 : meta,
|
2560
|
-
collectionType
|
2561
|
-
};
|
2562
|
-
return /* @__PURE__ */ jsxs(Flex, { direction: "column", gap: 2, width: "100%", children: [
|
2601
|
+
const plugins = useStrapiApp("HeaderToolbar", (state) => state.plugins);
|
2602
|
+
return /* @__PURE__ */ jsxs(Flex, { gap: 2, children: [
|
2563
2603
|
/* @__PURE__ */ jsx(
|
2564
2604
|
DescriptionComponentRenderer,
|
2565
2605
|
{
|
2566
|
-
props
|
2567
|
-
|
2568
|
-
|
2606
|
+
props: {
|
2607
|
+
activeTab: status,
|
2608
|
+
model,
|
2609
|
+
documentId: id,
|
2610
|
+
document: isCloning ? void 0 : document,
|
2611
|
+
meta: isCloning ? void 0 : meta,
|
2612
|
+
collectionType
|
2613
|
+
},
|
2614
|
+
descriptions: plugins["content-manager"].apis.getHeaderActions(),
|
2615
|
+
children: (actions2) => {
|
2616
|
+
if (actions2.length > 0) {
|
2617
|
+
return /* @__PURE__ */ jsx(HeaderActions, { actions: actions2 });
|
2618
|
+
} else {
|
2619
|
+
return null;
|
2620
|
+
}
|
2621
|
+
}
|
2569
2622
|
}
|
2570
2623
|
),
|
2571
|
-
/* @__PURE__ */ jsx(
|
2624
|
+
/* @__PURE__ */ jsx(
|
2625
|
+
DescriptionComponentRenderer,
|
2626
|
+
{
|
2627
|
+
props: {
|
2628
|
+
activeTab: status,
|
2629
|
+
model,
|
2630
|
+
documentId: id,
|
2631
|
+
document: isCloning ? void 0 : document,
|
2632
|
+
meta: isCloning ? void 0 : meta,
|
2633
|
+
collectionType
|
2634
|
+
},
|
2635
|
+
descriptions: plugins["content-manager"].apis.getDocumentActions(),
|
2636
|
+
children: (actions2) => {
|
2637
|
+
const headerActions = actions2.filter((action) => {
|
2638
|
+
const positions = Array.isArray(action.position) ? action.position : [action.position];
|
2639
|
+
return positions.includes("header");
|
2640
|
+
});
|
2641
|
+
return /* @__PURE__ */ jsx(
|
2642
|
+
DocumentActionsMenu,
|
2643
|
+
{
|
2644
|
+
actions: headerActions,
|
2645
|
+
label: formatMessage({
|
2646
|
+
id: "content-manager.containers.edit.header.more-actions",
|
2647
|
+
defaultMessage: "More actions"
|
2648
|
+
}),
|
2649
|
+
children: /* @__PURE__ */ jsx(Information, { activeTab: status })
|
2650
|
+
}
|
2651
|
+
);
|
2652
|
+
}
|
2653
|
+
}
|
2654
|
+
)
|
2572
2655
|
] });
|
2573
2656
|
};
|
2574
|
-
const
|
2575
|
-
|
2576
|
-
|
2657
|
+
const Information = ({ activeTab }) => {
|
2658
|
+
const { formatMessage } = useIntl();
|
2659
|
+
const { document, meta } = useDoc();
|
2660
|
+
if (!document || !document.id) {
|
2661
|
+
return null;
|
2662
|
+
}
|
2663
|
+
const createAndUpdateDocument = activeTab === "draft" ? document : meta?.availableStatus.find((status) => status.publishedAt === null);
|
2664
|
+
const publishDocument = activeTab === "published" ? document : meta?.availableStatus.find((status) => status.publishedAt !== null);
|
2665
|
+
const creator = createAndUpdateDocument?.[CREATED_BY_ATTRIBUTE_NAME] ? getDisplayName(createAndUpdateDocument[CREATED_BY_ATTRIBUTE_NAME]) : null;
|
2666
|
+
const updator = createAndUpdateDocument?.[UPDATED_BY_ATTRIBUTE_NAME] ? getDisplayName(createAndUpdateDocument[UPDATED_BY_ATTRIBUTE_NAME]) : null;
|
2667
|
+
const information = [
|
2577
2668
|
{
|
2578
|
-
|
2579
|
-
|
2580
|
-
|
2581
|
-
|
2582
|
-
borderColor: "neutral150",
|
2583
|
-
hasRadius: true,
|
2584
|
-
paddingBottom: 4,
|
2585
|
-
paddingLeft: 4,
|
2586
|
-
paddingRight: 4,
|
2587
|
-
paddingTop: 4,
|
2588
|
-
shadow: "tableShadow",
|
2589
|
-
gap: 3,
|
2590
|
-
direction: "column",
|
2591
|
-
justifyContent: "stretch",
|
2592
|
-
alignItems: "flex-start",
|
2593
|
-
children: [
|
2594
|
-
/* @__PURE__ */ jsx(Typography, { tag: "h2", variant: "sigma", textTransform: "uppercase", children: title }),
|
2595
|
-
children
|
2596
|
-
]
|
2597
|
-
}
|
2598
|
-
);
|
2599
|
-
});
|
2600
|
-
const HOOKS = {
|
2601
|
-
/**
|
2602
|
-
* Hook that allows to mutate the displayed headers of the list view table
|
2603
|
-
* @constant
|
2604
|
-
* @type {string}
|
2605
|
-
*/
|
2606
|
-
INJECT_COLUMN_IN_TABLE: "Admin/CM/pages/ListView/inject-column-in-table",
|
2607
|
-
/**
|
2608
|
-
* Hook that allows to mutate the CM's collection types links pre-set filters
|
2609
|
-
* @constant
|
2610
|
-
* @type {string}
|
2611
|
-
*/
|
2612
|
-
MUTATE_COLLECTION_TYPES_LINKS: "Admin/CM/pages/App/mutate-collection-types-links",
|
2613
|
-
/**
|
2614
|
-
* Hook that allows to mutate the CM's edit view layout
|
2615
|
-
* @constant
|
2616
|
-
* @type {string}
|
2617
|
-
*/
|
2618
|
-
MUTATE_EDIT_VIEW_LAYOUT: "Admin/CM/pages/EditView/mutate-edit-view-layout",
|
2619
|
-
/**
|
2620
|
-
* Hook that allows to mutate the CM's single types links pre-set filters
|
2621
|
-
* @constant
|
2622
|
-
* @type {string}
|
2623
|
-
*/
|
2624
|
-
MUTATE_SINGLE_TYPES_LINKS: "Admin/CM/pages/App/mutate-single-types-links"
|
2625
|
-
};
|
2626
|
-
const contentTypesApi = contentManagerApi.injectEndpoints({
|
2627
|
-
endpoints: (builder) => ({
|
2628
|
-
getContentTypeConfiguration: builder.query({
|
2629
|
-
query: (uid) => ({
|
2630
|
-
url: `/content-manager/content-types/${uid}/configuration`,
|
2631
|
-
method: "GET"
|
2669
|
+
isDisplayed: !!publishDocument?.[PUBLISHED_AT_ATTRIBUTE_NAME],
|
2670
|
+
label: formatMessage({
|
2671
|
+
id: "content-manager.containers.edit.information.last-published.label",
|
2672
|
+
defaultMessage: "Published"
|
2632
2673
|
}),
|
2633
|
-
|
2634
|
-
|
2635
|
-
|
2636
|
-
|
2637
|
-
|
2638
|
-
|
2639
|
-
|
2640
|
-
|
2641
|
-
|
2642
|
-
|
2643
|
-
|
2644
|
-
|
2645
|
-
|
2646
|
-
|
2647
|
-
|
2648
|
-
|
2674
|
+
value: formatMessage(
|
2675
|
+
{
|
2676
|
+
id: "content-manager.containers.edit.information.last-published.value",
|
2677
|
+
defaultMessage: `{time}{isAnonymous, select, true {} other { by {author}}}`
|
2678
|
+
},
|
2679
|
+
{
|
2680
|
+
time: /* @__PURE__ */ jsx(RelativeTime, { timestamp: new Date(publishDocument?.[PUBLISHED_AT_ATTRIBUTE_NAME]) }),
|
2681
|
+
isAnonymous: !publishDocument?.[PUBLISHED_BY_ATTRIBUTE_NAME],
|
2682
|
+
author: publishDocument?.[PUBLISHED_BY_ATTRIBUTE_NAME] ? getDisplayName(publishDocument?.[PUBLISHED_BY_ATTRIBUTE_NAME]) : null
|
2683
|
+
}
|
2684
|
+
)
|
2685
|
+
},
|
2686
|
+
{
|
2687
|
+
isDisplayed: !!createAndUpdateDocument?.[UPDATED_AT_ATTRIBUTE_NAME],
|
2688
|
+
label: formatMessage({
|
2689
|
+
id: "content-manager.containers.edit.information.last-draft.label",
|
2690
|
+
defaultMessage: "Updated"
|
2691
|
+
}),
|
2692
|
+
value: formatMessage(
|
2693
|
+
{
|
2694
|
+
id: "content-manager.containers.edit.information.last-draft.value",
|
2695
|
+
defaultMessage: `{time}{isAnonymous, select, true {} other { by {author}}}`
|
2696
|
+
},
|
2697
|
+
{
|
2698
|
+
time: /* @__PURE__ */ jsx(
|
2699
|
+
RelativeTime,
|
2700
|
+
{
|
2701
|
+
timestamp: new Date(createAndUpdateDocument?.[UPDATED_AT_ATTRIBUTE_NAME])
|
2702
|
+
}
|
2703
|
+
),
|
2704
|
+
isAnonymous: !updator,
|
2705
|
+
author: updator
|
2706
|
+
}
|
2707
|
+
)
|
2708
|
+
},
|
2709
|
+
{
|
2710
|
+
isDisplayed: !!createAndUpdateDocument?.[CREATED_AT_ATTRIBUTE_NAME],
|
2711
|
+
label: formatMessage({
|
2712
|
+
id: "content-manager.containers.edit.information.document.label",
|
2713
|
+
defaultMessage: "Created"
|
2649
2714
|
}),
|
2650
|
-
|
2651
|
-
|
2652
|
-
|
2653
|
-
|
2654
|
-
|
2655
|
-
{
|
2656
|
-
|
2657
|
-
|
2658
|
-
|
2659
|
-
|
2660
|
-
|
2661
|
-
|
2662
|
-
|
2663
|
-
|
2664
|
-
}
|
2665
|
-
|
2666
|
-
|
2667
|
-
|
2668
|
-
|
2669
|
-
|
2670
|
-
|
2671
|
-
|
2672
|
-
|
2673
|
-
|
2674
|
-
|
2675
|
-
|
2676
|
-
|
2677
|
-
|
2678
|
-
|
2715
|
+
value: formatMessage(
|
2716
|
+
{
|
2717
|
+
id: "content-manager.containers.edit.information.document.value",
|
2718
|
+
defaultMessage: `{time}{isAnonymous, select, true {} other { by {author}}}`
|
2719
|
+
},
|
2720
|
+
{
|
2721
|
+
time: /* @__PURE__ */ jsx(
|
2722
|
+
RelativeTime,
|
2723
|
+
{
|
2724
|
+
timestamp: new Date(createAndUpdateDocument?.[CREATED_AT_ATTRIBUTE_NAME])
|
2725
|
+
}
|
2726
|
+
),
|
2727
|
+
isAnonymous: !creator,
|
2728
|
+
author: creator
|
2729
|
+
}
|
2730
|
+
)
|
2731
|
+
}
|
2732
|
+
].filter((info) => info.isDisplayed);
|
2733
|
+
return /* @__PURE__ */ jsx(
|
2734
|
+
Flex,
|
2735
|
+
{
|
2736
|
+
borderWidth: "1px 0 0 0",
|
2737
|
+
borderStyle: "solid",
|
2738
|
+
borderColor: "neutral150",
|
2739
|
+
direction: "column",
|
2740
|
+
marginTop: 2,
|
2741
|
+
tag: "dl",
|
2742
|
+
padding: 5,
|
2743
|
+
gap: 3,
|
2744
|
+
alignItems: "flex-start",
|
2745
|
+
marginLeft: "-0.4rem",
|
2746
|
+
marginRight: "-0.4rem",
|
2747
|
+
width: "calc(100% + 8px)",
|
2748
|
+
children: information.map((info) => /* @__PURE__ */ jsxs(Flex, { gap: 1, direction: "column", alignItems: "flex-start", children: [
|
2749
|
+
/* @__PURE__ */ jsx(Typography, { tag: "dt", variant: "pi", fontWeight: "bold", children: info.label }),
|
2750
|
+
/* @__PURE__ */ jsx(Typography, { tag: "dd", variant: "pi", textColor: "neutral600", children: info.value })
|
2751
|
+
] }, info.label))
|
2752
|
+
}
|
2679
2753
|
);
|
2680
|
-
return {
|
2681
|
-
name: mainFieldName,
|
2682
|
-
type: mainFieldType ?? "string"
|
2683
|
-
};
|
2684
|
-
};
|
2685
|
-
const DEFAULT_SETTINGS = {
|
2686
|
-
bulkable: false,
|
2687
|
-
filterable: false,
|
2688
|
-
searchable: false,
|
2689
|
-
pagination: false,
|
2690
|
-
defaultSortBy: "",
|
2691
|
-
defaultSortOrder: "asc",
|
2692
|
-
mainField: "id",
|
2693
|
-
pageSize: 10
|
2694
2754
|
};
|
2695
|
-
const
|
2696
|
-
const
|
2697
|
-
const
|
2698
|
-
|
2699
|
-
|
2700
|
-
|
2701
|
-
|
2702
|
-
|
2703
|
-
|
2704
|
-
|
2705
|
-
error,
|
2706
|
-
isFetching: isFetchingConfigs
|
2707
|
-
} = useGetContentTypeConfigurationQuery(model);
|
2708
|
-
const isLoading = isLoadingSchemas || isFetchingConfigs || isLoadingConfigs;
|
2709
|
-
React.useEffect(() => {
|
2710
|
-
if (error) {
|
2711
|
-
toggleNotification({
|
2712
|
-
type: "danger",
|
2713
|
-
message: formatAPIError(error)
|
2714
|
-
});
|
2755
|
+
const HeaderActions = ({ actions: actions2 }) => {
|
2756
|
+
const [dialogId, setDialogId] = React.useState(null);
|
2757
|
+
const handleClick = (action) => async (e) => {
|
2758
|
+
if (!("options" in action)) {
|
2759
|
+
const { onClick = () => false, dialog, id } = action;
|
2760
|
+
const muteDialog = await onClick(e);
|
2761
|
+
if (dialog && !muteDialog) {
|
2762
|
+
e.preventDefault();
|
2763
|
+
setDialogId(id);
|
2764
|
+
}
|
2715
2765
|
}
|
2716
|
-
}, [error, formatAPIError, toggleNotification]);
|
2717
|
-
const editLayout = React.useMemo(
|
2718
|
-
() => data && !isLoading ? formatEditLayout(data, { schemas, schema, components }) : {
|
2719
|
-
layout: [],
|
2720
|
-
components: {},
|
2721
|
-
metadatas: {},
|
2722
|
-
options: {},
|
2723
|
-
settings: DEFAULT_SETTINGS
|
2724
|
-
},
|
2725
|
-
[data, isLoading, schemas, schema, components]
|
2726
|
-
);
|
2727
|
-
const listLayout = React.useMemo(() => {
|
2728
|
-
return data && !isLoading ? formatListLayout(data, { schemas, schema, components }) : {
|
2729
|
-
layout: [],
|
2730
|
-
metadatas: {},
|
2731
|
-
options: {},
|
2732
|
-
settings: DEFAULT_SETTINGS
|
2733
|
-
};
|
2734
|
-
}, [data, isLoading, schemas, schema, components]);
|
2735
|
-
const { layout: edit } = React.useMemo(
|
2736
|
-
() => runHookWaterfall(HOOKS.MUTATE_EDIT_VIEW_LAYOUT, {
|
2737
|
-
layout: editLayout,
|
2738
|
-
query
|
2739
|
-
}),
|
2740
|
-
[editLayout, query, runHookWaterfall]
|
2741
|
-
);
|
2742
|
-
return {
|
2743
|
-
error,
|
2744
|
-
isLoading,
|
2745
|
-
edit,
|
2746
|
-
list: listLayout
|
2747
2766
|
};
|
2748
|
-
|
2749
|
-
|
2750
|
-
|
2751
|
-
return
|
2752
|
-
|
2753
|
-
|
2754
|
-
|
2755
|
-
|
2756
|
-
|
2757
|
-
|
2758
|
-
|
2759
|
-
|
2760
|
-
|
2761
|
-
|
2762
|
-
|
2763
|
-
|
2764
|
-
schemas
|
2765
|
-
).reduce((panels, row) => {
|
2766
|
-
if (row.some((field) => field.type === "dynamiczone")) {
|
2767
|
-
panels.push([row]);
|
2768
|
-
currentPanelIndex += 2;
|
2767
|
+
const handleClose = () => {
|
2768
|
+
setDialogId(null);
|
2769
|
+
};
|
2770
|
+
return /* @__PURE__ */ jsx(Flex, { gap: 1, children: actions2.map((action) => {
|
2771
|
+
if (action.options) {
|
2772
|
+
return /* @__PURE__ */ jsx(
|
2773
|
+
SingleSelect,
|
2774
|
+
{
|
2775
|
+
size: "S",
|
2776
|
+
onChange: action.onSelect,
|
2777
|
+
"aria-label": action.label,
|
2778
|
+
...action,
|
2779
|
+
children: action.options.map(({ label, ...option }) => /* @__PURE__ */ jsx(SingleSelectOption, { ...option, children: label }, option.value))
|
2780
|
+
},
|
2781
|
+
action.id
|
2782
|
+
);
|
2769
2783
|
} else {
|
2770
|
-
if (
|
2771
|
-
|
2784
|
+
if (action.type === "icon") {
|
2785
|
+
return /* @__PURE__ */ jsxs(React.Fragment, { children: [
|
2786
|
+
/* @__PURE__ */ jsx(
|
2787
|
+
IconButton,
|
2788
|
+
{
|
2789
|
+
disabled: action.disabled,
|
2790
|
+
label: action.label,
|
2791
|
+
size: "S",
|
2792
|
+
onClick: handleClick(action),
|
2793
|
+
children: action.icon
|
2794
|
+
}
|
2795
|
+
),
|
2796
|
+
action.dialog ? /* @__PURE__ */ jsx(
|
2797
|
+
HeaderActionDialog,
|
2798
|
+
{
|
2799
|
+
...action.dialog,
|
2800
|
+
isOpen: dialogId === action.id,
|
2801
|
+
onClose: handleClose
|
2802
|
+
}
|
2803
|
+
) : null
|
2804
|
+
] }, action.id);
|
2772
2805
|
}
|
2773
|
-
panels[currentPanelIndex].push(row);
|
2774
2806
|
}
|
2775
|
-
|
2776
|
-
|
2777
|
-
|
2778
|
-
|
2779
|
-
|
2780
|
-
|
2781
|
-
|
2782
|
-
|
2783
|
-
|
2784
|
-
|
2785
|
-
|
2786
|
-
|
2787
|
-
|
2788
|
-
|
2789
|
-
|
2790
|
-
|
2791
|
-
|
2792
|
-
}
|
2793
|
-
|
2794
|
-
|
2795
|
-
|
2796
|
-
|
2797
|
-
|
2798
|
-
|
2799
|
-
|
2800
|
-
|
2807
|
+
}) });
|
2808
|
+
};
|
2809
|
+
const HeaderActionDialog = ({
|
2810
|
+
onClose,
|
2811
|
+
onCancel,
|
2812
|
+
title,
|
2813
|
+
content: Content,
|
2814
|
+
isOpen
|
2815
|
+
}) => {
|
2816
|
+
const handleClose = async () => {
|
2817
|
+
if (onCancel) {
|
2818
|
+
await onCancel();
|
2819
|
+
}
|
2820
|
+
onClose();
|
2821
|
+
};
|
2822
|
+
return /* @__PURE__ */ jsx(Dialog.Root, { open: isOpen, onOpenChange: handleClose, children: /* @__PURE__ */ jsxs(Dialog.Content, { children: [
|
2823
|
+
/* @__PURE__ */ jsx(Dialog.Header, { children: title }),
|
2824
|
+
typeof Content === "function" ? /* @__PURE__ */ jsx(Content, { onClose: handleClose }) : Content
|
2825
|
+
] }) });
|
2826
|
+
};
|
2827
|
+
const ConfigureTheViewAction = ({ collectionType, model }) => {
|
2828
|
+
const navigate = useNavigate();
|
2829
|
+
const { formatMessage } = useIntl();
|
2830
|
+
return {
|
2831
|
+
label: formatMessage({
|
2832
|
+
id: "app.links.configure-view",
|
2833
|
+
defaultMessage: "Configure the view"
|
2834
|
+
}),
|
2835
|
+
icon: /* @__PURE__ */ jsx(ListPlus, {}),
|
2836
|
+
onClick: () => {
|
2837
|
+
navigate(`../${collectionType}/${model}/configurations/edit`);
|
2801
2838
|
},
|
2802
|
-
|
2803
|
-
|
2839
|
+
position: "header"
|
2840
|
+
};
|
2841
|
+
};
|
2842
|
+
ConfigureTheViewAction.type = "configure-the-view";
|
2843
|
+
const EditTheModelAction = ({ model }) => {
|
2844
|
+
const navigate = useNavigate();
|
2845
|
+
const { formatMessage } = useIntl();
|
2804
2846
|
return {
|
2805
|
-
|
2806
|
-
|
2807
|
-
|
2808
|
-
|
2809
|
-
|
2810
|
-
|
2847
|
+
label: formatMessage({
|
2848
|
+
id: "content-manager.link-to-ctb",
|
2849
|
+
defaultMessage: "Edit the model"
|
2850
|
+
}),
|
2851
|
+
icon: /* @__PURE__ */ jsx(Pencil, {}),
|
2852
|
+
onClick: () => {
|
2853
|
+
navigate(`/plugins/content-type-builder/content-types/${model}`);
|
2811
2854
|
},
|
2812
|
-
|
2813
|
-
...schema?.options,
|
2814
|
-
...schema?.pluginOptions,
|
2815
|
-
...data.contentType.options
|
2816
|
-
}
|
2855
|
+
position: "header"
|
2817
2856
|
};
|
2818
2857
|
};
|
2819
|
-
|
2820
|
-
|
2821
|
-
|
2822
|
-
|
2823
|
-
|
2824
|
-
|
2858
|
+
EditTheModelAction.type = "edit-the-model";
|
2859
|
+
const DeleteAction$1 = ({ documentId, model, collectionType, document }) => {
|
2860
|
+
const navigate = useNavigate();
|
2861
|
+
const { formatMessage } = useIntl();
|
2862
|
+
const listViewPathMatch = useMatch(LIST_PATH);
|
2863
|
+
const canDelete = useDocumentRBAC("DeleteAction", (state) => state.canDelete);
|
2864
|
+
const { delete: deleteAction } = useDocumentActions();
|
2865
|
+
const { toggleNotification } = useNotification();
|
2866
|
+
const setSubmitting = useForm("DeleteAction", (state) => state.setSubmitting);
|
2867
|
+
const isLocalized = document?.locale != null;
|
2868
|
+
return {
|
2869
|
+
disabled: !canDelete || !document,
|
2870
|
+
label: formatMessage(
|
2871
|
+
{
|
2872
|
+
id: "content-manager.actions.delete.label",
|
2873
|
+
defaultMessage: "Delete entry{isLocalized, select, true { (all locales)} other {}}"
|
2874
|
+
},
|
2875
|
+
{ isLocalized }
|
2876
|
+
),
|
2877
|
+
icon: /* @__PURE__ */ jsx(Trash, {}),
|
2878
|
+
dialog: {
|
2879
|
+
type: "dialog",
|
2880
|
+
title: formatMessage({
|
2881
|
+
id: "app.components.ConfirmDialog.title",
|
2882
|
+
defaultMessage: "Confirmation"
|
2883
|
+
}),
|
2884
|
+
content: /* @__PURE__ */ jsxs(Flex, { direction: "column", gap: 2, children: [
|
2885
|
+
/* @__PURE__ */ jsx(WarningCircle, { width: "24px", height: "24px", fill: "danger600" }),
|
2886
|
+
/* @__PURE__ */ jsx(Typography, { tag: "p", variant: "omega", textAlign: "center", children: formatMessage({
|
2887
|
+
id: "content-manager.actions.delete.dialog.body",
|
2888
|
+
defaultMessage: "Are you sure?"
|
2889
|
+
}) })
|
2890
|
+
] }),
|
2891
|
+
onConfirm: async () => {
|
2892
|
+
if (!listViewPathMatch) {
|
2893
|
+
setSubmitting(true);
|
2894
|
+
}
|
2895
|
+
try {
|
2896
|
+
if (!documentId && collectionType !== SINGLE_TYPES) {
|
2897
|
+
console.error(
|
2898
|
+
"You're trying to delete a document without an id, this is likely a bug with Strapi. Please open an issue."
|
2899
|
+
);
|
2900
|
+
toggleNotification({
|
2901
|
+
message: formatMessage({
|
2902
|
+
id: "content-manager.actions.delete.error",
|
2903
|
+
defaultMessage: "An error occurred while trying to delete the document."
|
2904
|
+
}),
|
2905
|
+
type: "danger"
|
2906
|
+
});
|
2907
|
+
return;
|
2908
|
+
}
|
2909
|
+
const res = await deleteAction({
|
2910
|
+
documentId,
|
2911
|
+
model,
|
2912
|
+
collectionType,
|
2913
|
+
params: {
|
2914
|
+
locale: "*"
|
2915
|
+
}
|
2916
|
+
});
|
2917
|
+
if (!("error" in res)) {
|
2918
|
+
navigate({ pathname: `../${collectionType}/${model}` }, { replace: true });
|
2919
|
+
}
|
2920
|
+
} finally {
|
2921
|
+
if (!listViewPathMatch) {
|
2922
|
+
setSubmitting(false);
|
2923
|
+
}
|
2924
|
+
}
|
2825
2925
|
}
|
2826
|
-
const { edit: metadata } = metadatas[field.name];
|
2827
|
-
const settings = attribute.type === "component" && components ? components.configurations[attribute.component].settings : {};
|
2828
|
-
return {
|
2829
|
-
attribute,
|
2830
|
-
disabled: !metadata.editable,
|
2831
|
-
hint: metadata.description,
|
2832
|
-
label: metadata.label ?? "",
|
2833
|
-
name: field.name,
|
2834
|
-
// @ts-expect-error – mainField does exist on the metadata for a relation.
|
2835
|
-
mainField: getMainField(attribute, metadata.mainField || settings.mainField, {
|
2836
|
-
schemas,
|
2837
|
-
components: components?.schemas ?? {}
|
2838
|
-
}),
|
2839
|
-
placeholder: metadata.placeholder ?? "",
|
2840
|
-
required: attribute.required ?? false,
|
2841
|
-
size: field.size,
|
2842
|
-
unique: "unique" in attribute ? attribute.unique : false,
|
2843
|
-
visible: metadata.visible ?? true,
|
2844
|
-
type: attribute.type
|
2845
|
-
};
|
2846
|
-
}).filter((field) => field !== null)
|
2847
|
-
);
|
2848
|
-
};
|
2849
|
-
const formatListLayout = (data, {
|
2850
|
-
schemas,
|
2851
|
-
schema,
|
2852
|
-
components
|
2853
|
-
}) => {
|
2854
|
-
const listMetadatas = Object.entries(data.contentType.metadatas).reduce(
|
2855
|
-
(acc, [attribute, metadata]) => {
|
2856
|
-
return {
|
2857
|
-
...acc,
|
2858
|
-
[attribute]: metadata.list
|
2859
|
-
};
|
2860
2926
|
},
|
2861
|
-
|
2862
|
-
|
2863
|
-
|
2864
|
-
|
2865
|
-
|
2866
|
-
|
2867
|
-
|
2868
|
-
|
2869
|
-
|
2870
|
-
|
2871
|
-
|
2872
|
-
|
2873
|
-
|
2874
|
-
|
2875
|
-
|
2876
|
-
|
2877
|
-
|
2927
|
+
variant: "danger",
|
2928
|
+
position: ["header", "table-row"]
|
2929
|
+
};
|
2930
|
+
};
|
2931
|
+
DeleteAction$1.type = "delete";
|
2932
|
+
const DEFAULT_HEADER_ACTIONS = [EditTheModelAction, ConfigureTheViewAction, DeleteAction$1];
|
2933
|
+
const Panels = () => {
|
2934
|
+
const isCloning = useMatch(CLONE_PATH) !== null;
|
2935
|
+
const [
|
2936
|
+
{
|
2937
|
+
query: { status }
|
2938
|
+
}
|
2939
|
+
] = useQueryParams({
|
2940
|
+
status: "draft"
|
2941
|
+
});
|
2942
|
+
const { model, id, document, meta, collectionType } = useDoc();
|
2943
|
+
const plugins = useStrapiApp("Panels", (state) => state.plugins);
|
2944
|
+
const props = {
|
2945
|
+
activeTab: status,
|
2946
|
+
model,
|
2947
|
+
documentId: id,
|
2948
|
+
document: isCloning ? void 0 : document,
|
2949
|
+
meta: isCloning ? void 0 : meta,
|
2950
|
+
collectionType
|
2951
|
+
};
|
2952
|
+
return /* @__PURE__ */ jsx(Flex, { direction: "column", alignItems: "stretch", gap: 2, children: /* @__PURE__ */ jsx(
|
2953
|
+
DescriptionComponentRenderer,
|
2954
|
+
{
|
2955
|
+
props,
|
2956
|
+
descriptions: plugins["content-manager"].apis.getEditViewSidePanels(),
|
2957
|
+
children: (panels) => panels.map(({ content, id: id2, ...description }) => /* @__PURE__ */ jsx(Panel, { ...description, children: content }, id2))
|
2878
2958
|
}
|
2959
|
+
) });
|
2960
|
+
};
|
2961
|
+
const ActionsPanel = () => {
|
2962
|
+
const { formatMessage } = useIntl();
|
2963
|
+
return {
|
2964
|
+
title: formatMessage({
|
2965
|
+
id: "content-manager.containers.edit.panels.default.title",
|
2966
|
+
defaultMessage: "Entry"
|
2967
|
+
}),
|
2968
|
+
content: /* @__PURE__ */ jsx(ActionsPanelContent, {})
|
2879
2969
|
};
|
2880
2970
|
};
|
2881
|
-
|
2882
|
-
|
2883
|
-
|
2884
|
-
|
2885
|
-
|
2971
|
+
ActionsPanel.type = "actions";
|
2972
|
+
const ActionsPanelContent = () => {
|
2973
|
+
const isCloning = useMatch(CLONE_PATH) !== null;
|
2974
|
+
const [
|
2975
|
+
{
|
2976
|
+
query: { status = "draft" }
|
2886
2977
|
}
|
2887
|
-
|
2888
|
-
|
2889
|
-
|
2890
|
-
|
2891
|
-
|
2892
|
-
|
2893
|
-
|
2894
|
-
|
2895
|
-
|
2896
|
-
|
2897
|
-
|
2898
|
-
|
2899
|
-
|
2900
|
-
|
2978
|
+
] = useQueryParams();
|
2979
|
+
const { model, id, document, meta, collectionType } = useDoc();
|
2980
|
+
const plugins = useStrapiApp("ActionsPanel", (state) => state.plugins);
|
2981
|
+
const props = {
|
2982
|
+
activeTab: status,
|
2983
|
+
model,
|
2984
|
+
documentId: id,
|
2985
|
+
document: isCloning ? void 0 : document,
|
2986
|
+
meta: isCloning ? void 0 : meta,
|
2987
|
+
collectionType
|
2988
|
+
};
|
2989
|
+
return /* @__PURE__ */ jsxs(Flex, { direction: "column", gap: 2, width: "100%", children: [
|
2990
|
+
/* @__PURE__ */ jsx(
|
2991
|
+
DescriptionComponentRenderer,
|
2992
|
+
{
|
2993
|
+
props,
|
2994
|
+
descriptions: plugins["content-manager"].apis.getDocumentActions(),
|
2995
|
+
children: (actions2) => /* @__PURE__ */ jsx(DocumentActions, { actions: actions2 })
|
2996
|
+
}
|
2997
|
+
),
|
2998
|
+
/* @__PURE__ */ jsx(InjectionZone, { area: "editView.right-links", slug: model })
|
2999
|
+
] });
|
2901
3000
|
};
|
3001
|
+
const Panel = React.forwardRef(({ children, title }, ref) => {
|
3002
|
+
return /* @__PURE__ */ jsxs(
|
3003
|
+
Flex,
|
3004
|
+
{
|
3005
|
+
ref,
|
3006
|
+
tag: "aside",
|
3007
|
+
"aria-labelledby": "additional-information",
|
3008
|
+
background: "neutral0",
|
3009
|
+
borderColor: "neutral150",
|
3010
|
+
hasRadius: true,
|
3011
|
+
paddingBottom: 4,
|
3012
|
+
paddingLeft: 4,
|
3013
|
+
paddingRight: 4,
|
3014
|
+
paddingTop: 4,
|
3015
|
+
shadow: "tableShadow",
|
3016
|
+
gap: 3,
|
3017
|
+
direction: "column",
|
3018
|
+
justifyContent: "stretch",
|
3019
|
+
alignItems: "flex-start",
|
3020
|
+
children: [
|
3021
|
+
/* @__PURE__ */ jsx(Typography, { tag: "h2", variant: "sigma", textTransform: "uppercase", textColor: "neutral600", children: title }),
|
3022
|
+
children
|
3023
|
+
]
|
3024
|
+
}
|
3025
|
+
);
|
3026
|
+
});
|
2902
3027
|
const ConfirmBulkActionDialog = ({
|
2903
3028
|
onToggleDialog,
|
2904
3029
|
isOpen = false,
|
@@ -3876,6 +4001,15 @@ const { setInitialData } = actions;
|
|
3876
4001
|
const reducer = combineReducers({
|
3877
4002
|
app: reducer$1
|
3878
4003
|
});
|
4004
|
+
const FEATURE_ID = "preview";
|
4005
|
+
const previewAdmin = {
|
4006
|
+
bootstrap(app) {
|
4007
|
+
if (!window.strapi.future.isEnabled(FEATURE_ID)) {
|
4008
|
+
return {};
|
4009
|
+
}
|
4010
|
+
console.log("Bootstrapping preview admin");
|
4011
|
+
}
|
4012
|
+
};
|
3879
4013
|
const index = {
|
3880
4014
|
register(app) {
|
3881
4015
|
const cm = new ContentManagerPlugin();
|
@@ -3895,7 +4029,7 @@ const index = {
|
|
3895
4029
|
app.router.addRoute({
|
3896
4030
|
path: "content-manager/*",
|
3897
4031
|
lazy: async () => {
|
3898
|
-
const { Layout } = await import("./layout-
|
4032
|
+
const { Layout } = await import("./layout-C-eg7m9Y.mjs");
|
3899
4033
|
return {
|
3900
4034
|
Component: Layout
|
3901
4035
|
};
|
@@ -3908,11 +4042,14 @@ const index = {
|
|
3908
4042
|
if (typeof historyAdmin.bootstrap === "function") {
|
3909
4043
|
historyAdmin.bootstrap(app);
|
3910
4044
|
}
|
4045
|
+
if (typeof previewAdmin.bootstrap === "function") {
|
4046
|
+
previewAdmin.bootstrap(app);
|
4047
|
+
}
|
3911
4048
|
},
|
3912
4049
|
async registerTrads({ locales }) {
|
3913
4050
|
const importedTrads = await Promise.all(
|
3914
4051
|
locales.map((locale) => {
|
3915
|
-
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-
|
4052
|
+
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-DKV44jRb.mjs"), "./translations/es.json": () => import("./es-CeXiYflN.mjs"), "./translations/eu.json": () => import("./eu-CdALomew.mjs"), "./translations/fr.json": () => import("./fr-DxrTDLl6.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 }) => {
|
3916
4053
|
return {
|
3917
4054
|
data: prefixPluginTranslations(data, PLUGIN_ID),
|
3918
4055
|
locale
|
@@ -3940,7 +4077,8 @@ export {
|
|
3940
4077
|
InjectionZone as I,
|
3941
4078
|
useDocument as J,
|
3942
4079
|
index as K,
|
3943
|
-
|
4080
|
+
useContentManagerContext as L,
|
4081
|
+
useDocumentActions as M,
|
3944
4082
|
Panels as P,
|
3945
4083
|
RelativeTime as R,
|
3946
4084
|
SINGLE_TYPES as S,
|
@@ -3972,4 +4110,4 @@ export {
|
|
3972
4110
|
capitalise as y,
|
3973
4111
|
useUpdateContentTypeConfigurationMutation as z
|
3974
4112
|
};
|
3975
|
-
//# sourceMappingURL=index-
|
4113
|
+
//# sourceMappingURL=index-DUZzV11o.mjs.map
|