@contentful/field-editor-reference 5.9.0 → 5.10.0
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/cjs/__fixtures__/FakeSdk.js +183 -0
- package/dist/cjs/__fixtures__/asset/index.js +37 -0
- package/dist/cjs/__fixtures__/content-type/index.js +16 -0
- package/dist/cjs/__fixtures__/entry/index.js +33 -0
- package/dist/cjs/__fixtures__/fixtures.js +71 -0
- package/dist/cjs/__fixtures__/locale/index.js +40 -0
- package/dist/cjs/__fixtures__/space/index.js +16 -0
- package/dist/cjs/assets/MultipleMediaEditor.js +86 -0
- package/dist/cjs/assets/SingleMediaEditor.js +69 -0
- package/dist/cjs/assets/WrappedAssetCard/AssetCardActions.js +125 -0
- package/dist/cjs/assets/WrappedAssetCard/FetchingWrappedAssetCard.js +171 -0
- package/dist/cjs/assets/WrappedAssetCard/WrappedAssetCard.js +159 -0
- package/dist/cjs/assets/WrappedAssetCard/WrappedAssetLink.js +130 -0
- package/dist/cjs/assets/index.js +24 -0
- package/dist/cjs/common/EntityStore.js +420 -0
- package/dist/cjs/common/MultipleReferenceEditor.js +164 -0
- package/dist/cjs/common/ReferenceEditor.js +74 -0
- package/dist/cjs/common/SingleReferenceEditor.js +118 -0
- package/dist/cjs/common/SortableLinkList.js +95 -0
- package/dist/cjs/common/customCardTypes.js +44 -0
- package/dist/cjs/common/useAccessApi.js +19 -0
- package/dist/cjs/common/useContentTypePermissions.js +54 -0
- package/dist/cjs/common/useEditorPermissions.js +77 -0
- package/dist/cjs/common/useEditorPermissions.spec.js +205 -0
- package/dist/cjs/components/AssetThumbnail/AssetThumbnail.js +62 -0
- package/dist/cjs/components/CreateEntryLinkButton/CreateEntryLinkButton.js +102 -0
- package/dist/cjs/components/CreateEntryLinkButton/CreateEntryLinkButton.spec.js +254 -0
- package/dist/cjs/components/CreateEntryLinkButton/CreateEntryMenuTrigger.js +199 -0
- package/dist/cjs/components/CreateEntryLinkButton/CreateEntryMenuTrigger.spec.js +190 -0
- package/dist/cjs/components/CreateEntryLinkButton/useGlobalMouseUp.js +19 -0
- package/dist/cjs/components/LinkActions/CombinedLinkActions.js +167 -0
- package/dist/cjs/components/LinkActions/LinkActions.js +123 -0
- package/dist/cjs/components/LinkActions/LinkEntityActions.js +186 -0
- package/dist/cjs/components/LinkActions/NoLinkPermissionsInfo.js +54 -0
- package/dist/cjs/components/LinkActions/helpers.js +78 -0
- package/dist/cjs/components/LinkActions/redesignStyles.js +44 -0
- package/dist/cjs/components/LinkActions/styles.js +33 -0
- package/dist/cjs/components/MissingEntityCard/MissingEntityCard.js +75 -0
- package/dist/cjs/components/MissingEntityCard/styles.js +29 -0
- package/dist/cjs/components/ScheduledIconWithTooltip/ScheduleTooltip.js +75 -0
- package/dist/cjs/components/ScheduledIconWithTooltip/ScheduledIconWithTooltip.js +81 -0
- package/dist/cjs/components/ScheduledIconWithTooltip/formatDateAndTime.js +45 -0
- package/dist/cjs/components/SpaceName/SpaceName.js +91 -0
- package/dist/cjs/components/index.js +44 -0
- package/dist/cjs/entries/MultipleEntryReferenceEditor.js +86 -0
- package/dist/cjs/entries/SingleEntryReferenceEditor.js +74 -0
- package/dist/cjs/entries/WrappedEntryCard/FetchingWrappedEntryCard.js +189 -0
- package/dist/cjs/entries/WrappedEntryCard/WrappedEntryCard.js +181 -0
- package/dist/cjs/entries/index.js +24 -0
- package/dist/cjs/index.js +92 -0
- package/dist/cjs/resources/Cards/ContentfulEntryCard.js +87 -0
- package/dist/cjs/resources/Cards/ResourceCard.js +111 -0
- package/dist/cjs/resources/Cards/UnsupportedEntityCard.js +64 -0
- package/dist/cjs/resources/MultipleResourceReferenceEditor.js +157 -0
- package/dist/cjs/resources/MultipleResourceReferenceEditor.spec.js +297 -0
- package/dist/cjs/resources/SingleResourceReferenceEditor.js +87 -0
- package/dist/cjs/resources/SingleResourceReferenceEditor.spec.js +161 -0
- package/dist/cjs/resources/index.js +19 -0
- package/dist/cjs/resources/testHelpers/resourceEditorHelpers.js +121 -0
- package/dist/cjs/resources/useResourceLinkActions.js +88 -0
- package/dist/cjs/types.js +22 -0
- package/dist/cjs/utils/fromFieldValidations.js +54 -0
- package/dist/esm/__fixtures__/FakeSdk.js +173 -0
- package/dist/esm/__fixtures__/asset/index.js +6 -0
- package/dist/esm/__fixtures__/content-type/index.js +2 -0
- package/dist/esm/__fixtures__/entry/index.js +5 -0
- package/dist/esm/__fixtures__/fixtures.js +6 -0
- package/dist/esm/__fixtures__/locale/index.js +15 -0
- package/dist/esm/__fixtures__/space/index.js +2 -0
- package/dist/esm/assets/MultipleMediaEditor.js +37 -0
- package/dist/esm/assets/SingleMediaEditor.js +20 -0
- package/dist/esm/assets/WrappedAssetCard/AssetCardActions.js +63 -0
- package/dist/esm/assets/WrappedAssetCard/FetchingWrappedAssetCard.js +122 -0
- package/dist/esm/assets/WrappedAssetCard/WrappedAssetCard.js +105 -0
- package/dist/esm/assets/WrappedAssetCard/WrappedAssetLink.js +76 -0
- package/dist/esm/assets/index.js +3 -0
- package/dist/esm/common/EntityStore.js +347 -0
- package/dist/esm/common/MultipleReferenceEditor.js +111 -0
- package/dist/esm/common/ReferenceEditor.js +20 -0
- package/dist/esm/common/SingleReferenceEditor.js +70 -0
- package/dist/esm/common/SortableLinkList.js +41 -0
- package/dist/esm/common/customCardTypes.js +1 -0
- package/dist/esm/common/useAccessApi.js +9 -0
- package/dist/esm/common/useContentTypePermissions.js +44 -0
- package/dist/esm/common/useEditorPermissions.js +67 -0
- package/dist/esm/common/useEditorPermissions.spec.js +201 -0
- package/dist/esm/components/AssetThumbnail/AssetThumbnail.js +13 -0
- package/dist/esm/components/CreateEntryLinkButton/CreateEntryLinkButton.js +48 -0
- package/dist/esm/components/CreateEntryLinkButton/CreateEntryLinkButton.spec.js +206 -0
- package/dist/esm/components/CreateEntryLinkButton/CreateEntryMenuTrigger.js +145 -0
- package/dist/esm/components/CreateEntryLinkButton/CreateEntryMenuTrigger.spec.js +142 -0
- package/dist/esm/components/CreateEntryLinkButton/useGlobalMouseUp.js +9 -0
- package/dist/esm/components/LinkActions/CombinedLinkActions.js +118 -0
- package/dist/esm/components/LinkActions/LinkActions.js +66 -0
- package/dist/esm/components/LinkActions/LinkEntityActions.js +127 -0
- package/dist/esm/components/LinkActions/NoLinkPermissionsInfo.js +5 -0
- package/dist/esm/components/LinkActions/helpers.js +57 -0
- package/dist/esm/components/LinkActions/redesignStyles.js +18 -0
- package/dist/esm/components/LinkActions/styles.js +10 -0
- package/dist/esm/components/MissingEntityCard/MissingEntityCard.js +26 -0
- package/dist/esm/components/MissingEntityCard/styles.js +11 -0
- package/dist/esm/components/ScheduledIconWithTooltip/ScheduleTooltip.js +18 -0
- package/dist/esm/components/ScheduledIconWithTooltip/ScheduledIconWithTooltip.js +32 -0
- package/dist/esm/components/ScheduledIconWithTooltip/formatDateAndTime.js +19 -0
- package/dist/esm/components/SpaceName/SpaceName.js +37 -0
- package/dist/esm/components/index.js +8 -0
- package/dist/esm/entries/MultipleEntryReferenceEditor.js +37 -0
- package/dist/esm/entries/SingleEntryReferenceEditor.js +25 -0
- package/dist/esm/entries/WrappedEntryCard/FetchingWrappedEntryCard.js +135 -0
- package/dist/esm/entries/WrappedEntryCard/WrappedEntryCard.js +127 -0
- package/dist/esm/entries/index.js +3 -0
- package/dist/esm/index.js +7 -0
- package/dist/esm/resources/Cards/ContentfulEntryCard.js +38 -0
- package/dist/esm/resources/Cards/ResourceCard.js +62 -0
- package/dist/esm/resources/Cards/UnsupportedEntityCard.js +15 -0
- package/dist/esm/resources/MultipleResourceReferenceEditor.js +104 -0
- package/dist/esm/resources/MultipleResourceReferenceEditor.spec.js +254 -0
- package/dist/esm/resources/SingleResourceReferenceEditor.js +33 -0
- package/dist/esm/resources/SingleResourceReferenceEditor.spec.js +118 -0
- package/dist/esm/resources/index.js +2 -0
- package/dist/esm/resources/testHelpers/resourceEditorHelpers.js +103 -0
- package/dist/esm/resources/useResourceLinkActions.js +78 -0
- package/dist/esm/types.js +1 -0
- package/dist/esm/utils/fromFieldValidations.js +39 -0
- package/dist/{__fixtures__ → types/__fixtures__}/FakeSdk.d.ts +8 -8
- package/dist/{__fixtures__ → types/__fixtures__}/asset/index.d.ts +6 -6
- package/dist/{__fixtures__ → types/__fixtures__}/content-type/index.d.ts +2 -2
- package/dist/{__fixtures__ → types/__fixtures__}/entry/index.d.ts +5 -5
- package/dist/{__fixtures__ → types/__fixtures__}/fixtures.d.ts +6 -6
- package/dist/{__fixtures__ → types/__fixtures__}/locale/index.d.ts +42 -42
- package/dist/{__fixtures__ → types/__fixtures__}/space/index.d.ts +2 -2
- package/dist/{assets → types/assets}/MultipleMediaEditor.d.ts +10 -10
- package/dist/types/assets/SingleMediaEditor.d.ts +10 -0
- package/dist/{assets → types/assets}/WrappedAssetCard/AssetCardActions.d.ts +11 -11
- package/dist/{assets → types/assets}/WrappedAssetCard/FetchingWrappedAssetCard.d.ts +17 -17
- package/dist/{assets → types/assets}/WrappedAssetCard/WrappedAssetCard.d.ts +24 -24
- package/dist/{assets → types/assets}/WrappedAssetCard/WrappedAssetLink.d.ts +16 -16
- package/dist/{assets → types/assets}/index.d.ts +3 -3
- package/dist/{common → types/common}/EntityStore.d.ts +62 -62
- package/dist/{common → types/common}/MultipleReferenceEditor.d.ts +25 -25
- package/dist/{common → types/common}/ReferenceEditor.d.ts +46 -46
- package/dist/{common → types/common}/SingleReferenceEditor.d.ts +24 -24
- package/dist/{common → types/common}/SortableLinkList.d.ts +19 -19
- package/dist/{common → types/common}/customCardTypes.d.ts +29 -29
- package/dist/types/common/useAccessApi.d.ts +16 -0
- package/dist/{common → types/common}/useContentTypePermissions.d.ts +17 -17
- package/dist/{common → types/common}/useEditorPermissions.d.ts +17 -17
- package/dist/types/common/useEditorPermissions.spec.d.ts +1 -0
- package/dist/{components → types/components}/AssetThumbnail/AssetThumbnail.d.ts +7 -7
- package/dist/{components → types/components}/CreateEntryLinkButton/CreateEntryLinkButton.d.ts +19 -19
- package/dist/types/components/CreateEntryLinkButton/CreateEntryLinkButton.spec.d.ts +1 -0
- package/dist/{components → types/components}/CreateEntryLinkButton/CreateEntryMenuTrigger.d.ts +31 -31
- package/dist/types/components/CreateEntryLinkButton/CreateEntryMenuTrigger.spec.d.ts +1 -0
- package/dist/{components → types/components}/CreateEntryLinkButton/useGlobalMouseUp.d.ts +1 -1
- package/dist/{components → types/components}/LinkActions/CombinedLinkActions.d.ts +10 -10
- package/dist/{components → types/components}/LinkActions/LinkActions.d.ts +26 -26
- package/dist/{components → types/components}/LinkActions/LinkEntityActions.d.ts +24 -24
- package/dist/types/components/LinkActions/NoLinkPermissionsInfo.d.ts +2 -0
- package/dist/{components → types/components}/LinkActions/helpers.d.ts +26 -26
- package/dist/{components → types/components}/LinkActions/redesignStyles.d.ts +3 -3
- package/dist/{components → types/components}/LinkActions/styles.d.ts +2 -2
- package/dist/{components → types/components}/MissingEntityCard/MissingEntityCard.d.ts +8 -8
- package/dist/{components → types/components}/MissingEntityCard/styles.d.ts +2 -2
- package/dist/{components → types/components}/ScheduledIconWithTooltip/ScheduleTooltip.d.ts +11 -11
- package/dist/{components → types/components}/ScheduledIconWithTooltip/ScheduledIconWithTooltip.d.ts +10 -10
- package/dist/{components → types/components}/ScheduledIconWithTooltip/formatDateAndTime.d.ts +15 -15
- package/dist/types/components/SpaceName/SpaceName.d.ts +6 -0
- package/dist/{components → types/components}/index.d.ts +9 -9
- package/dist/{entries → types/entries}/MultipleEntryReferenceEditor.d.ts +3 -3
- package/dist/{entries → types/entries}/SingleEntryReferenceEditor.d.ts +8 -8
- package/dist/{entries → types/entries}/WrappedEntryCard/FetchingWrappedEntryCard.d.ts +18 -18
- package/dist/{entries → types/entries}/WrappedEntryCard/WrappedEntryCard.d.ts +35 -35
- package/dist/{entries → types/entries}/index.d.ts +3 -3
- package/dist/{index.d.ts → types/index.d.ts} +9 -8
- package/dist/{resources → types/resources}/Cards/ContentfulEntryCard.d.ts +21 -21
- package/dist/{resources → types/resources}/Cards/ResourceCard.d.ts +12 -12
- package/dist/{resources → types/resources}/Cards/UnsupportedEntityCard.d.ts +4 -4
- package/dist/{resources → types/resources}/MultipleResourceReferenceEditor.d.ts +7 -7
- package/dist/types/resources/MultipleResourceReferenceEditor.spec.d.ts +1 -0
- package/dist/{resources → types/resources}/SingleResourceReferenceEditor.d.ts +7 -7
- package/dist/types/resources/SingleResourceReferenceEditor.spec.d.ts +1 -0
- package/dist/{resources → types/resources}/index.d.ts +2 -2
- package/dist/{resources → types/resources}/testHelpers/resourceEditorHelpers.d.ts +50 -50
- package/dist/{resources → types/resources}/useResourceLinkActions.d.ts +7 -7
- package/dist/{types.d.ts → types/types.d.ts} +104 -104
- package/dist/{utils → types/utils}/fromFieldValidations.d.ts +21 -21
- package/package.json +25 -11
- package/CHANGELOG.md +0 -860
- package/dist/assets/SingleMediaEditor.d.ts +0 -10
- package/dist/common/useAccessApi.d.ts +0 -16
- package/dist/components/LinkActions/NoLinkPermissionsInfo.d.ts +0 -2
- package/dist/components/SpaceName/SpaceName.d.ts +0 -6
- package/dist/field-editor-reference.cjs.development.js +0 -2753
- package/dist/field-editor-reference.cjs.development.js.map +0 -1
- package/dist/field-editor-reference.cjs.production.min.js +0 -2
- package/dist/field-editor-reference.cjs.production.min.js.map +0 -1
- package/dist/field-editor-reference.esm.js +0 -2727
- package/dist/field-editor-reference.esm.js.map +0 -1
- package/dist/index.js +0 -8
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import { useMemo } from 'react';
|
|
3
|
+
import { CombinedLinkActions } from './CombinedLinkActions';
|
|
4
|
+
import { createEntity, selectMultipleEntities, selectSingleEntity } from './helpers';
|
|
5
|
+
import { LinkActions } from './LinkActions';
|
|
6
|
+
export function useLinkActionsProps(props) {
|
|
7
|
+
const { sdk , editorPermissions , entityType , canLinkMultiple , isDisabled , actionLabels , itemsLength } = props;
|
|
8
|
+
const maxLinksCount = editorPermissions.validations.numberOfLinks?.max;
|
|
9
|
+
const value = sdk.field.getValue();
|
|
10
|
+
const linkCount = Array.isArray(value) ? value.length : value ? 1 : 0;
|
|
11
|
+
const isFull = !!maxLinksCount && maxLinksCount <= linkCount;
|
|
12
|
+
const isEmpty = linkCount === 0;
|
|
13
|
+
const onCreated = React.useCallback((entity, index = itemsLength, slide)=>{
|
|
14
|
+
props.onCreate(entity.sys.id, index);
|
|
15
|
+
props.onAction && props.onAction({
|
|
16
|
+
type: 'create_and_link',
|
|
17
|
+
entity: entityType,
|
|
18
|
+
entityData: entity,
|
|
19
|
+
slide,
|
|
20
|
+
index
|
|
21
|
+
});
|
|
22
|
+
}, [
|
|
23
|
+
entityType,
|
|
24
|
+
props.onCreate,
|
|
25
|
+
props.onAction
|
|
26
|
+
]);
|
|
27
|
+
const onLinkedExisting = React.useCallback((entities, index = itemsLength)=>{
|
|
28
|
+
props.onLink(entities.map((item)=>item.sys.id), index);
|
|
29
|
+
entities.forEach((entity, i)=>{
|
|
30
|
+
props.onAction && props.onAction({
|
|
31
|
+
type: 'select_and_link',
|
|
32
|
+
entity: entityType,
|
|
33
|
+
entityData: entity,
|
|
34
|
+
index: index === undefined ? undefined : index + i
|
|
35
|
+
});
|
|
36
|
+
});
|
|
37
|
+
}, [
|
|
38
|
+
entityType,
|
|
39
|
+
props.onLink,
|
|
40
|
+
props.onAction
|
|
41
|
+
]);
|
|
42
|
+
const onCreate = React.useCallback(async (contentTypeId, index)=>{
|
|
43
|
+
const { entity , slide } = await createEntity({
|
|
44
|
+
sdk,
|
|
45
|
+
entityType,
|
|
46
|
+
contentTypeId
|
|
47
|
+
});
|
|
48
|
+
if (!entity) {
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
51
|
+
onCreated(entity, index, slide);
|
|
52
|
+
}, [
|
|
53
|
+
sdk,
|
|
54
|
+
entityType,
|
|
55
|
+
onCreated
|
|
56
|
+
]);
|
|
57
|
+
const onLinkExisting = React.useCallback(async (index)=>{
|
|
58
|
+
const entity = await selectSingleEntity({
|
|
59
|
+
sdk,
|
|
60
|
+
entityType,
|
|
61
|
+
editorPermissions
|
|
62
|
+
});
|
|
63
|
+
if (!entity) {
|
|
64
|
+
return;
|
|
65
|
+
}
|
|
66
|
+
onLinkedExisting([
|
|
67
|
+
entity
|
|
68
|
+
], index);
|
|
69
|
+
}, [
|
|
70
|
+
sdk,
|
|
71
|
+
entityType,
|
|
72
|
+
onLinkedExisting
|
|
73
|
+
]);
|
|
74
|
+
const onLinkSeveralExisting = React.useCallback(async (index)=>{
|
|
75
|
+
const entities = await selectMultipleEntities({
|
|
76
|
+
sdk,
|
|
77
|
+
entityType,
|
|
78
|
+
editorPermissions
|
|
79
|
+
});
|
|
80
|
+
if (!entities || entities.length === 0) {
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
onLinkedExisting(entities, index);
|
|
84
|
+
}, [
|
|
85
|
+
sdk,
|
|
86
|
+
entityType,
|
|
87
|
+
onLinkedExisting
|
|
88
|
+
]);
|
|
89
|
+
return useMemo(()=>({
|
|
90
|
+
entityType,
|
|
91
|
+
canLinkMultiple,
|
|
92
|
+
isDisabled,
|
|
93
|
+
isEmpty,
|
|
94
|
+
isFull,
|
|
95
|
+
canCreateEntity: editorPermissions.canCreateEntity,
|
|
96
|
+
canLinkEntity: editorPermissions.canLinkEntity,
|
|
97
|
+
contentTypes: editorPermissions.creatableContentTypes,
|
|
98
|
+
onCreate,
|
|
99
|
+
onLinkExisting: canLinkMultiple ? onLinkSeveralExisting : onLinkExisting,
|
|
100
|
+
actionLabels,
|
|
101
|
+
onCreated,
|
|
102
|
+
onLinkedExisting,
|
|
103
|
+
itemsLength
|
|
104
|
+
}), [
|
|
105
|
+
entityType,
|
|
106
|
+
canLinkMultiple,
|
|
107
|
+
isDisabled,
|
|
108
|
+
isEmpty,
|
|
109
|
+
isFull,
|
|
110
|
+
editorPermissions.canCreateEntity,
|
|
111
|
+
editorPermissions.canLinkEntity,
|
|
112
|
+
actionLabels,
|
|
113
|
+
editorPermissions.creatableContentTypes.map((ct)=>ct.sys.id).join(':'),
|
|
114
|
+
onCreate,
|
|
115
|
+
onLinkExisting,
|
|
116
|
+
onLinkSeveralExisting,
|
|
117
|
+
onCreated,
|
|
118
|
+
onLinkedExisting,
|
|
119
|
+
itemsLength
|
|
120
|
+
]);
|
|
121
|
+
}
|
|
122
|
+
export function LinkEntityActions({ renderCustomActions , ...props }) {
|
|
123
|
+
return renderCustomActions ? renderCustomActions(props) : React.createElement(LinkActions, props);
|
|
124
|
+
}
|
|
125
|
+
export function CombinedLinkEntityActions({ renderCustomActions , ...props }) {
|
|
126
|
+
return renderCustomActions ? renderCustomActions(props) : React.createElement(CombinedLinkActions, props);
|
|
127
|
+
}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import { Paragraph } from '@contentful/f36-components';
|
|
3
|
+
export function NoLinkPermissionsInfo() {
|
|
4
|
+
return React.createElement(Paragraph, null, "You don't have permission to view this content or this field is not correctly configured. Contact your administrator for help.");
|
|
5
|
+
}
|
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
const getContentTypeIds = (contentTypes)=>contentTypes.map((ct)=>ct.sys.id);
|
|
2
|
+
export async function createEntity(props) {
|
|
3
|
+
if (props.entityType === 'Entry') {
|
|
4
|
+
if (!props.contentTypeId) {
|
|
5
|
+
return {};
|
|
6
|
+
}
|
|
7
|
+
const { entity , slide } = await props.sdk.navigator.openNewEntry(props.contentTypeId, {
|
|
8
|
+
slideIn: true
|
|
9
|
+
});
|
|
10
|
+
return {
|
|
11
|
+
entity,
|
|
12
|
+
slide
|
|
13
|
+
};
|
|
14
|
+
} else {
|
|
15
|
+
const { entity , slide } = await props.sdk.navigator.openNewAsset({
|
|
16
|
+
slideIn: true
|
|
17
|
+
});
|
|
18
|
+
return {
|
|
19
|
+
entity,
|
|
20
|
+
slide
|
|
21
|
+
};
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
export async function selectSingleEntity(props) {
|
|
25
|
+
if (props.entityType === 'Entry') {
|
|
26
|
+
return await props.sdk.dialogs.selectSingleEntry({
|
|
27
|
+
locale: props.sdk.field.locale,
|
|
28
|
+
contentTypes: getContentTypeIds(props.editorPermissions.readableContentTypes)
|
|
29
|
+
});
|
|
30
|
+
} else {
|
|
31
|
+
return props.sdk.dialogs.selectSingleAsset({
|
|
32
|
+
locale: props.sdk.field.locale,
|
|
33
|
+
mimetypeGroups: props.editorPermissions.validations.mimetypeGroups
|
|
34
|
+
});
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
export async function selectMultipleEntities(props) {
|
|
38
|
+
const value = props.sdk.field.getValue();
|
|
39
|
+
const linkCount = Array.isArray(value) ? value.length : value ? 1 : 0;
|
|
40
|
+
const min = Math.max((props.editorPermissions.validations.numberOfLinks?.min || 1) - linkCount, 1);
|
|
41
|
+
const max = (props.editorPermissions.validations.numberOfLinks?.max || +Infinity) - linkCount;
|
|
42
|
+
if (props.entityType === 'Entry') {
|
|
43
|
+
return await props.sdk.dialogs.selectMultipleEntries({
|
|
44
|
+
locale: props.sdk.field.locale,
|
|
45
|
+
contentTypes: getContentTypeIds(props.editorPermissions.readableContentTypes),
|
|
46
|
+
min,
|
|
47
|
+
max
|
|
48
|
+
});
|
|
49
|
+
} else {
|
|
50
|
+
return props.sdk.dialogs.selectMultipleAssets({
|
|
51
|
+
locale: props.sdk.field.locale,
|
|
52
|
+
mimetypeGroups: props.editorPermissions.validations.mimetypeGroups,
|
|
53
|
+
min,
|
|
54
|
+
max
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import tokens from '@contentful/f36-tokens';
|
|
2
|
+
import { css } from 'emotion';
|
|
3
|
+
export const container = css({
|
|
4
|
+
display: 'flex',
|
|
5
|
+
border: `1px dashed ${tokens.gray500}`,
|
|
6
|
+
borderRadius: tokens.borderRadiusMedium,
|
|
7
|
+
justifyContent: 'center',
|
|
8
|
+
padding: tokens.spacingXl
|
|
9
|
+
});
|
|
10
|
+
export const action = css({
|
|
11
|
+
textDecoration: 'none',
|
|
12
|
+
fontWeight: 'bold'
|
|
13
|
+
});
|
|
14
|
+
export const chevronIcon = css({
|
|
15
|
+
float: 'right',
|
|
16
|
+
marginLeft: tokens.spacingXs,
|
|
17
|
+
marginRight: -tokens.spacing2Xs
|
|
18
|
+
});
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import { Card, Flex, IconButton, SectionHeading } from '@contentful/f36-components';
|
|
3
|
+
import { CloseIcon } from '@contentful/f36-icons';
|
|
4
|
+
import * as styles from './styles';
|
|
5
|
+
export function MissingEntityCard(props) {
|
|
6
|
+
return React.createElement(Card, {
|
|
7
|
+
className: styles.card,
|
|
8
|
+
testId: "cf-ui-missing-entry-card"
|
|
9
|
+
}, React.createElement(Flex, {
|
|
10
|
+
alignItems: "center",
|
|
11
|
+
justifyContent: "space-between"
|
|
12
|
+
}, React.createElement("div", {
|
|
13
|
+
className: props.asSquare ? styles.squareCard : ''
|
|
14
|
+
}, React.createElement(SectionHeading, {
|
|
15
|
+
marginBottom: "none"
|
|
16
|
+
}, props.entityType, " is missing or inaccessible")), !props.isDisabled && props.onRemove && React.createElement(IconButton, {
|
|
17
|
+
variant: "transparent",
|
|
18
|
+
icon: React.createElement(CloseIcon, {
|
|
19
|
+
variant: "muted"
|
|
20
|
+
}),
|
|
21
|
+
"aria-label": "Delete",
|
|
22
|
+
onClick: ()=>{
|
|
23
|
+
props.onRemove && props.onRemove();
|
|
24
|
+
}
|
|
25
|
+
})));
|
|
26
|
+
}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import { Tooltip } from '@contentful/f36-components';
|
|
3
|
+
import { formatDateAndTime } from './formatDateAndTime';
|
|
4
|
+
export const getScheduleTooltipContent = ({ job , jobsCount })=>{
|
|
5
|
+
return `Will ${job.action.toLowerCase()} ${formatDateAndTime(job.scheduledFor.datetime).toLowerCase()}
|
|
6
|
+
${jobsCount > 1 ? `+ ${jobsCount - 1} more` : ''}`;
|
|
7
|
+
};
|
|
8
|
+
export const ScheduleTooltip = ({ job , jobsCount , children })=>{
|
|
9
|
+
return React.createElement(Tooltip, {
|
|
10
|
+
placement: "top",
|
|
11
|
+
testId: job.sys.id,
|
|
12
|
+
as: "div",
|
|
13
|
+
content: getScheduleTooltipContent({
|
|
14
|
+
job,
|
|
15
|
+
jobsCount
|
|
16
|
+
})
|
|
17
|
+
}, children);
|
|
18
|
+
};
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import { ScheduleTooltip } from './ScheduleTooltip';
|
|
3
|
+
export const ScheduledIconWithTooltip = ({ entityType , entityId , getEntityScheduledActions , children })=>{
|
|
4
|
+
const [status, setStatus] = React.useState({
|
|
5
|
+
type: 'loading'
|
|
6
|
+
});
|
|
7
|
+
React.useEffect(()=>{
|
|
8
|
+
getEntityScheduledActions(entityType, entityId).then((data)=>{
|
|
9
|
+
setStatus({
|
|
10
|
+
type: 'loaded',
|
|
11
|
+
jobs: data
|
|
12
|
+
});
|
|
13
|
+
}).catch((e)=>{
|
|
14
|
+
setStatus({
|
|
15
|
+
type: 'error',
|
|
16
|
+
error: e
|
|
17
|
+
});
|
|
18
|
+
});
|
|
19
|
+
}, []);
|
|
20
|
+
if (status.type === 'loading' || status.type === 'error') {
|
|
21
|
+
return null;
|
|
22
|
+
}
|
|
23
|
+
const jobs = status.jobs ?? [];
|
|
24
|
+
if (jobs.length === 0) {
|
|
25
|
+
return null;
|
|
26
|
+
}
|
|
27
|
+
const mostRelevantJob = jobs[0];
|
|
28
|
+
return React.createElement(ScheduleTooltip, {
|
|
29
|
+
job: mostRelevantJob,
|
|
30
|
+
jobsCount: jobs.length
|
|
31
|
+
}, children);
|
|
32
|
+
};
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import moment from 'moment';
|
|
2
|
+
export const formatDate = (date, short)=>{
|
|
3
|
+
switch(moment().startOf('day').diff(moment(date).startOf('day'), 'days')){
|
|
4
|
+
case 0:
|
|
5
|
+
return short ? 'Today' : `Today, ${moment(date).format('DD MMM YYYY')}`;
|
|
6
|
+
case -1:
|
|
7
|
+
return short ? 'Tomorrow' : `Tomorrow, ${moment(date).format('DD MMM YYYY')}`;
|
|
8
|
+
case 1:
|
|
9
|
+
return short ? 'Yesterday' : `Yesterday, ${moment(date).format('DD MMM YYYY')}`;
|
|
10
|
+
default:
|
|
11
|
+
return moment(date).format('ddd, DD MMM YYYY');
|
|
12
|
+
}
|
|
13
|
+
};
|
|
14
|
+
export const formatTime = (date)=>{
|
|
15
|
+
return moment.utc(date).local().format('h:mm A');
|
|
16
|
+
};
|
|
17
|
+
export const formatDateAndTime = (date, short)=>{
|
|
18
|
+
return `${formatDate(date, short)} at ${formatTime(date)}`;
|
|
19
|
+
};
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import { Flex, Text, Tooltip } from '@contentful/f36-components';
|
|
3
|
+
import { FolderOpenTrimmedIcon } from '@contentful/f36-icons';
|
|
4
|
+
import tokens from '@contentful/f36-tokens';
|
|
5
|
+
import { css } from 'emotion';
|
|
6
|
+
const styles = {
|
|
7
|
+
spaceIcon: css({
|
|
8
|
+
flexShrink: 0,
|
|
9
|
+
fill: tokens.purple600
|
|
10
|
+
}),
|
|
11
|
+
spaceName: css({
|
|
12
|
+
color: tokens.gray700,
|
|
13
|
+
fontSize: tokens.fontSizeS,
|
|
14
|
+
fontWeight: tokens.fontWeightDemiBold,
|
|
15
|
+
maxWidth: '80px',
|
|
16
|
+
textOverflow: 'ellipsis',
|
|
17
|
+
overflow: 'hidden',
|
|
18
|
+
whiteSpace: 'nowrap'
|
|
19
|
+
})
|
|
20
|
+
};
|
|
21
|
+
export function SpaceName(props) {
|
|
22
|
+
return React.createElement(Tooltip, {
|
|
23
|
+
placement: "top",
|
|
24
|
+
as: "div",
|
|
25
|
+
content: `Space: ${props.spaceName}`
|
|
26
|
+
}, React.createElement(Flex, {
|
|
27
|
+
alignItems: "center",
|
|
28
|
+
gap: "spacingXs",
|
|
29
|
+
marginRight: "spacingS"
|
|
30
|
+
}, React.createElement(FolderOpenTrimmedIcon, {
|
|
31
|
+
className: styles.spaceIcon,
|
|
32
|
+
size: "tiny",
|
|
33
|
+
"aria-label": "Source space"
|
|
34
|
+
}), React.createElement(Text, {
|
|
35
|
+
className: styles.spaceName
|
|
36
|
+
}, props.spaceName)));
|
|
37
|
+
}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
export { CombinedLinkActions } from './LinkActions/CombinedLinkActions';
|
|
2
|
+
export { MissingEntityCard } from './MissingEntityCard/MissingEntityCard';
|
|
3
|
+
export { LinkEntityActions } from './LinkActions/LinkEntityActions';
|
|
4
|
+
export { CreateEntryLinkButton } from './CreateEntryLinkButton/CreateEntryLinkButton';
|
|
5
|
+
export { CreateEntryMenuTrigger } from './CreateEntryLinkButton/CreateEntryMenuTrigger';
|
|
6
|
+
export { ScheduledIconWithTooltip } from './ScheduledIconWithTooltip/ScheduledIconWithTooltip';
|
|
7
|
+
export { getScheduleTooltipContent } from './ScheduledIconWithTooltip/ScheduleTooltip';
|
|
8
|
+
export { AssetThumbnail } from './AssetThumbnail/AssetThumbnail';
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import { MultipleReferenceEditor } from '../common/MultipleReferenceEditor';
|
|
3
|
+
import { SortableLinkList } from '../common/SortableLinkList';
|
|
4
|
+
import { FetchingWrappedEntryCard } from './WrappedEntryCard/FetchingWrappedEntryCard';
|
|
5
|
+
export function MultipleEntryReferenceEditor(props) {
|
|
6
|
+
const [indexToUpdate, setIndexToUpdate] = React.useState(undefined);
|
|
7
|
+
const updateBeforeSortStart = ({ index })=>{
|
|
8
|
+
setIndexToUpdate(index);
|
|
9
|
+
};
|
|
10
|
+
return React.createElement(MultipleReferenceEditor, {
|
|
11
|
+
...props,
|
|
12
|
+
entityType: "Entry",
|
|
13
|
+
setIndexToUpdate: setIndexToUpdate
|
|
14
|
+
}, (childrenProps)=>React.createElement(SortableLinkList, {
|
|
15
|
+
...childrenProps,
|
|
16
|
+
axis: "y",
|
|
17
|
+
useDragHandle: true,
|
|
18
|
+
updateBeforeSortStart: updateBeforeSortStart
|
|
19
|
+
}, ({ items , item , index , isDisabled , DragHandle })=>{
|
|
20
|
+
const lastIndex = items.length - 1;
|
|
21
|
+
return React.createElement(FetchingWrappedEntryCard, {
|
|
22
|
+
...childrenProps,
|
|
23
|
+
key: `${item.sys.id}-${index}`,
|
|
24
|
+
index: index,
|
|
25
|
+
allContentTypes: childrenProps.allContentTypes,
|
|
26
|
+
isDisabled: isDisabled,
|
|
27
|
+
entryId: item.sys.id,
|
|
28
|
+
onRemove: ()=>{
|
|
29
|
+
childrenProps.setValue(items.filter((_value, i)=>i !== index));
|
|
30
|
+
},
|
|
31
|
+
onMoveTop: index !== 0 ? ()=>childrenProps.onMove(index, 0) : undefined,
|
|
32
|
+
onMoveBottom: index !== lastIndex ? ()=>childrenProps.onMove(index, lastIndex) : undefined,
|
|
33
|
+
renderDragHandle: DragHandle,
|
|
34
|
+
isBeingDragged: index === indexToUpdate
|
|
35
|
+
});
|
|
36
|
+
}));
|
|
37
|
+
}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import { SingleReferenceEditor } from '../common/SingleReferenceEditor';
|
|
3
|
+
import { FetchingWrappedEntryCard } from './WrappedEntryCard/FetchingWrappedEntryCard';
|
|
4
|
+
export function SingleEntryReferenceEditor(props) {
|
|
5
|
+
return React.createElement(SingleReferenceEditor, {
|
|
6
|
+
...props,
|
|
7
|
+
entityType: "Entry"
|
|
8
|
+
}, ({ allContentTypes , isDisabled , entityId , setValue , renderCustomCard , hasCardRemoveActions , hasCardEditActions })=>{
|
|
9
|
+
return React.createElement(FetchingWrappedEntryCard, {
|
|
10
|
+
...props,
|
|
11
|
+
allContentTypes: allContentTypes,
|
|
12
|
+
isDisabled: isDisabled,
|
|
13
|
+
entryId: entityId,
|
|
14
|
+
renderCustomCard: renderCustomCard,
|
|
15
|
+
hasCardEditActions: hasCardEditActions,
|
|
16
|
+
hasCardRemoveActions: hasCardRemoveActions,
|
|
17
|
+
onRemove: ()=>{
|
|
18
|
+
setValue(null);
|
|
19
|
+
}
|
|
20
|
+
});
|
|
21
|
+
});
|
|
22
|
+
}
|
|
23
|
+
SingleEntryReferenceEditor.defaultProps = {
|
|
24
|
+
isInitiallyDisabled: true
|
|
25
|
+
};
|
|
@@ -0,0 +1,135 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import { EntryCard } from '@contentful/f36-components';
|
|
3
|
+
import get from 'lodash/get';
|
|
4
|
+
import { useEntity, useEntityLoader } from '../../common/EntityStore';
|
|
5
|
+
import { MissingEntityCard } from '../../components';
|
|
6
|
+
import { WrappedEntryCard } from './WrappedEntryCard';
|
|
7
|
+
async function openEntry(sdk, entryId, options) {
|
|
8
|
+
let slide;
|
|
9
|
+
if (options.bulkEditing) {
|
|
10
|
+
try {
|
|
11
|
+
const result = await sdk.navigator.openBulkEditor(sdk.entry.getSys().id, {
|
|
12
|
+
fieldId: sdk.field.id,
|
|
13
|
+
locale: sdk.field.locale,
|
|
14
|
+
index: options.index ?? 0
|
|
15
|
+
});
|
|
16
|
+
slide = result.slide;
|
|
17
|
+
return slide;
|
|
18
|
+
} catch (e) {}
|
|
19
|
+
}
|
|
20
|
+
const result = await sdk.navigator.openEntry(entryId, {
|
|
21
|
+
slideIn: true
|
|
22
|
+
});
|
|
23
|
+
slide = result.slide;
|
|
24
|
+
return slide;
|
|
25
|
+
}
|
|
26
|
+
export function FetchingWrappedEntryCard(props) {
|
|
27
|
+
const { data: entry , status } = useEntity('Entry', props.entryId);
|
|
28
|
+
const { getEntityScheduledActions } = useEntityLoader();
|
|
29
|
+
const loadEntityScheduledActions = React.useCallback(()=>getEntityScheduledActions('Entry', props.entryId), [
|
|
30
|
+
getEntityScheduledActions,
|
|
31
|
+
props.entryId
|
|
32
|
+
]);
|
|
33
|
+
const size = props.viewType === 'link' ? 'small' : 'default';
|
|
34
|
+
const { getEntity } = useEntityLoader();
|
|
35
|
+
const getAsset = (assetId)=>getEntity('Asset', assetId);
|
|
36
|
+
const onEdit = async ()=>{
|
|
37
|
+
const slide = await openEntry(props.sdk, props.entryId, {
|
|
38
|
+
bulkEditing: props.parameters.instance.bulkEditing,
|
|
39
|
+
index: props.index
|
|
40
|
+
});
|
|
41
|
+
props.onAction && props.onAction({
|
|
42
|
+
entity: 'Entry',
|
|
43
|
+
type: 'edit',
|
|
44
|
+
id: props.entryId,
|
|
45
|
+
contentTypeId: get(entry, 'sys.contentType.sys.id'),
|
|
46
|
+
slide
|
|
47
|
+
});
|
|
48
|
+
};
|
|
49
|
+
const onRemoveEntry = ()=>{
|
|
50
|
+
props.onRemove();
|
|
51
|
+
props.onAction && props.onAction({
|
|
52
|
+
entity: 'Entry',
|
|
53
|
+
type: 'delete',
|
|
54
|
+
id: props.entryId,
|
|
55
|
+
contentTypeId: get(entry, 'sys.contentType.sys.id')
|
|
56
|
+
});
|
|
57
|
+
};
|
|
58
|
+
React.useEffect(()=>{
|
|
59
|
+
if (entry) {
|
|
60
|
+
props.onAction && props.onAction({
|
|
61
|
+
type: 'rendered',
|
|
62
|
+
entity: 'Entry'
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
}, [
|
|
66
|
+
entry
|
|
67
|
+
]);
|
|
68
|
+
return React.useMemo(()=>{
|
|
69
|
+
if (status === 'error') {
|
|
70
|
+
const card = React.createElement(MissingEntityCard, {
|
|
71
|
+
entityType: "Entry",
|
|
72
|
+
isDisabled: props.isDisabled,
|
|
73
|
+
onRemove: onRemoveEntry
|
|
74
|
+
});
|
|
75
|
+
if (props.renderCustomMissingEntityCard) {
|
|
76
|
+
return props.renderCustomMissingEntityCard({
|
|
77
|
+
defaultCard: card,
|
|
78
|
+
entity: {
|
|
79
|
+
id: props.entryId,
|
|
80
|
+
type: 'Entry'
|
|
81
|
+
}
|
|
82
|
+
});
|
|
83
|
+
}
|
|
84
|
+
return card;
|
|
85
|
+
}
|
|
86
|
+
if (status === 'loading') {
|
|
87
|
+
return React.createElement(EntryCard, {
|
|
88
|
+
size: size,
|
|
89
|
+
isLoading: true
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
const sharedCardProps = {
|
|
93
|
+
index: props.index,
|
|
94
|
+
entity: entry,
|
|
95
|
+
entityUrl: props.getEntityUrl && props.getEntityUrl(entry.sys.id),
|
|
96
|
+
contentType: props.allContentTypes.find((contentType)=>contentType.sys.id === entry.sys.contentType.sys.id),
|
|
97
|
+
isDisabled: props.isDisabled,
|
|
98
|
+
size,
|
|
99
|
+
localeCode: props.sdk.field.locale,
|
|
100
|
+
defaultLocaleCode: props.sdk.locales.default,
|
|
101
|
+
renderDragHandle: props.renderDragHandle,
|
|
102
|
+
onEdit,
|
|
103
|
+
onRemove: onRemoveEntry,
|
|
104
|
+
onMoveTop: props.onMoveTop,
|
|
105
|
+
onMoveBottom: props.onMoveBottom,
|
|
106
|
+
isBeingDragged: props.isBeingDragged
|
|
107
|
+
};
|
|
108
|
+
const { hasCardEditActions , hasCardMoveActions , hasCardRemoveActions } = props;
|
|
109
|
+
function renderDefaultCard(props) {
|
|
110
|
+
const builtinCardProps = {
|
|
111
|
+
...sharedCardProps,
|
|
112
|
+
...props,
|
|
113
|
+
hasCardEditActions,
|
|
114
|
+
hasCardMoveActions,
|
|
115
|
+
hasCardRemoveActions,
|
|
116
|
+
getAsset,
|
|
117
|
+
getEntityScheduledActions: loadEntityScheduledActions,
|
|
118
|
+
entry: (props?.entity) || sharedCardProps.entity,
|
|
119
|
+
entryUrl: props?.entityUrl || sharedCardProps.entityUrl
|
|
120
|
+
};
|
|
121
|
+
return React.createElement(WrappedEntryCard, builtinCardProps);
|
|
122
|
+
}
|
|
123
|
+
if (props.renderCustomCard) {
|
|
124
|
+
const renderedCustomCard = props.renderCustomCard(sharedCardProps, {}, renderDefaultCard);
|
|
125
|
+
if (renderedCustomCard !== false) {
|
|
126
|
+
return renderedCustomCard;
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
return renderDefaultCard();
|
|
130
|
+
}, [
|
|
131
|
+
props,
|
|
132
|
+
status,
|
|
133
|
+
entry
|
|
134
|
+
]);
|
|
135
|
+
}
|