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