@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.
Files changed (199) hide show
  1. package/dist/cjs/__fixtures__/FakeSdk.js +183 -0
  2. package/dist/cjs/__fixtures__/asset/index.js +37 -0
  3. package/dist/cjs/__fixtures__/content-type/index.js +16 -0
  4. package/dist/cjs/__fixtures__/entry/index.js +33 -0
  5. package/dist/cjs/__fixtures__/fixtures.js +71 -0
  6. package/dist/cjs/__fixtures__/locale/index.js +40 -0
  7. package/dist/cjs/__fixtures__/space/index.js +16 -0
  8. package/dist/cjs/assets/MultipleMediaEditor.js +86 -0
  9. package/dist/cjs/assets/SingleMediaEditor.js +69 -0
  10. package/dist/cjs/assets/WrappedAssetCard/AssetCardActions.js +125 -0
  11. package/dist/cjs/assets/WrappedAssetCard/FetchingWrappedAssetCard.js +171 -0
  12. package/dist/cjs/assets/WrappedAssetCard/WrappedAssetCard.js +159 -0
  13. package/dist/cjs/assets/WrappedAssetCard/WrappedAssetLink.js +130 -0
  14. package/dist/cjs/assets/index.js +24 -0
  15. package/dist/cjs/common/EntityStore.js +420 -0
  16. package/dist/cjs/common/MultipleReferenceEditor.js +164 -0
  17. package/dist/cjs/common/ReferenceEditor.js +74 -0
  18. package/dist/cjs/common/SingleReferenceEditor.js +118 -0
  19. package/dist/cjs/common/SortableLinkList.js +95 -0
  20. package/dist/cjs/common/customCardTypes.js +44 -0
  21. package/dist/cjs/common/useAccessApi.js +19 -0
  22. package/dist/cjs/common/useContentTypePermissions.js +54 -0
  23. package/dist/cjs/common/useEditorPermissions.js +77 -0
  24. package/dist/cjs/common/useEditorPermissions.spec.js +205 -0
  25. package/dist/cjs/components/AssetThumbnail/AssetThumbnail.js +62 -0
  26. package/dist/cjs/components/CreateEntryLinkButton/CreateEntryLinkButton.js +102 -0
  27. package/dist/cjs/components/CreateEntryLinkButton/CreateEntryLinkButton.spec.js +254 -0
  28. package/dist/cjs/components/CreateEntryLinkButton/CreateEntryMenuTrigger.js +199 -0
  29. package/dist/cjs/components/CreateEntryLinkButton/CreateEntryMenuTrigger.spec.js +190 -0
  30. package/dist/cjs/components/CreateEntryLinkButton/useGlobalMouseUp.js +19 -0
  31. package/dist/cjs/components/LinkActions/CombinedLinkActions.js +167 -0
  32. package/dist/cjs/components/LinkActions/LinkActions.js +123 -0
  33. package/dist/cjs/components/LinkActions/LinkEntityActions.js +186 -0
  34. package/dist/cjs/components/LinkActions/NoLinkPermissionsInfo.js +54 -0
  35. package/dist/cjs/components/LinkActions/helpers.js +78 -0
  36. package/dist/cjs/components/LinkActions/redesignStyles.js +44 -0
  37. package/dist/cjs/components/LinkActions/styles.js +33 -0
  38. package/dist/cjs/components/MissingEntityCard/MissingEntityCard.js +75 -0
  39. package/dist/cjs/components/MissingEntityCard/styles.js +29 -0
  40. package/dist/cjs/components/ScheduledIconWithTooltip/ScheduleTooltip.js +75 -0
  41. package/dist/cjs/components/ScheduledIconWithTooltip/ScheduledIconWithTooltip.js +81 -0
  42. package/dist/cjs/components/ScheduledIconWithTooltip/formatDateAndTime.js +45 -0
  43. package/dist/cjs/components/SpaceName/SpaceName.js +91 -0
  44. package/dist/cjs/components/index.js +44 -0
  45. package/dist/cjs/entries/MultipleEntryReferenceEditor.js +86 -0
  46. package/dist/cjs/entries/SingleEntryReferenceEditor.js +74 -0
  47. package/dist/cjs/entries/WrappedEntryCard/FetchingWrappedEntryCard.js +189 -0
  48. package/dist/cjs/entries/WrappedEntryCard/WrappedEntryCard.js +181 -0
  49. package/dist/cjs/entries/index.js +24 -0
  50. package/dist/cjs/index.js +92 -0
  51. package/dist/cjs/resources/Cards/ContentfulEntryCard.js +87 -0
  52. package/dist/cjs/resources/Cards/ResourceCard.js +111 -0
  53. package/dist/cjs/resources/Cards/UnsupportedEntityCard.js +64 -0
  54. package/dist/cjs/resources/MultipleResourceReferenceEditor.js +157 -0
  55. package/dist/cjs/resources/MultipleResourceReferenceEditor.spec.js +297 -0
  56. package/dist/cjs/resources/SingleResourceReferenceEditor.js +87 -0
  57. package/dist/cjs/resources/SingleResourceReferenceEditor.spec.js +161 -0
  58. package/dist/cjs/resources/index.js +19 -0
  59. package/dist/cjs/resources/testHelpers/resourceEditorHelpers.js +121 -0
  60. package/dist/cjs/resources/useResourceLinkActions.js +88 -0
  61. package/dist/cjs/types.js +22 -0
  62. package/dist/cjs/utils/fromFieldValidations.js +54 -0
  63. package/dist/esm/__fixtures__/FakeSdk.js +173 -0
  64. package/dist/esm/__fixtures__/asset/index.js +6 -0
  65. package/dist/esm/__fixtures__/content-type/index.js +2 -0
  66. package/dist/esm/__fixtures__/entry/index.js +5 -0
  67. package/dist/esm/__fixtures__/fixtures.js +6 -0
  68. package/dist/esm/__fixtures__/locale/index.js +15 -0
  69. package/dist/esm/__fixtures__/space/index.js +2 -0
  70. package/dist/esm/assets/MultipleMediaEditor.js +37 -0
  71. package/dist/esm/assets/SingleMediaEditor.js +20 -0
  72. package/dist/esm/assets/WrappedAssetCard/AssetCardActions.js +63 -0
  73. package/dist/esm/assets/WrappedAssetCard/FetchingWrappedAssetCard.js +122 -0
  74. package/dist/esm/assets/WrappedAssetCard/WrappedAssetCard.js +105 -0
  75. package/dist/esm/assets/WrappedAssetCard/WrappedAssetLink.js +76 -0
  76. package/dist/esm/assets/index.js +3 -0
  77. package/dist/esm/common/EntityStore.js +347 -0
  78. package/dist/esm/common/MultipleReferenceEditor.js +111 -0
  79. package/dist/esm/common/ReferenceEditor.js +20 -0
  80. package/dist/esm/common/SingleReferenceEditor.js +70 -0
  81. package/dist/esm/common/SortableLinkList.js +41 -0
  82. package/dist/esm/common/customCardTypes.js +1 -0
  83. package/dist/esm/common/useAccessApi.js +9 -0
  84. package/dist/esm/common/useContentTypePermissions.js +44 -0
  85. package/dist/esm/common/useEditorPermissions.js +67 -0
  86. package/dist/esm/common/useEditorPermissions.spec.js +201 -0
  87. package/dist/esm/components/AssetThumbnail/AssetThumbnail.js +13 -0
  88. package/dist/esm/components/CreateEntryLinkButton/CreateEntryLinkButton.js +48 -0
  89. package/dist/esm/components/CreateEntryLinkButton/CreateEntryLinkButton.spec.js +206 -0
  90. package/dist/esm/components/CreateEntryLinkButton/CreateEntryMenuTrigger.js +145 -0
  91. package/dist/esm/components/CreateEntryLinkButton/CreateEntryMenuTrigger.spec.js +142 -0
  92. package/dist/esm/components/CreateEntryLinkButton/useGlobalMouseUp.js +9 -0
  93. package/dist/esm/components/LinkActions/CombinedLinkActions.js +118 -0
  94. package/dist/esm/components/LinkActions/LinkActions.js +66 -0
  95. package/dist/esm/components/LinkActions/LinkEntityActions.js +127 -0
  96. package/dist/esm/components/LinkActions/NoLinkPermissionsInfo.js +5 -0
  97. package/dist/esm/components/LinkActions/helpers.js +57 -0
  98. package/dist/esm/components/LinkActions/redesignStyles.js +18 -0
  99. package/dist/esm/components/LinkActions/styles.js +10 -0
  100. package/dist/esm/components/MissingEntityCard/MissingEntityCard.js +26 -0
  101. package/dist/esm/components/MissingEntityCard/styles.js +11 -0
  102. package/dist/esm/components/ScheduledIconWithTooltip/ScheduleTooltip.js +18 -0
  103. package/dist/esm/components/ScheduledIconWithTooltip/ScheduledIconWithTooltip.js +32 -0
  104. package/dist/esm/components/ScheduledIconWithTooltip/formatDateAndTime.js +19 -0
  105. package/dist/esm/components/SpaceName/SpaceName.js +37 -0
  106. package/dist/esm/components/index.js +8 -0
  107. package/dist/esm/entries/MultipleEntryReferenceEditor.js +37 -0
  108. package/dist/esm/entries/SingleEntryReferenceEditor.js +25 -0
  109. package/dist/esm/entries/WrappedEntryCard/FetchingWrappedEntryCard.js +135 -0
  110. package/dist/esm/entries/WrappedEntryCard/WrappedEntryCard.js +127 -0
  111. package/dist/esm/entries/index.js +3 -0
  112. package/dist/esm/index.js +7 -0
  113. package/dist/esm/resources/Cards/ContentfulEntryCard.js +38 -0
  114. package/dist/esm/resources/Cards/ResourceCard.js +62 -0
  115. package/dist/esm/resources/Cards/UnsupportedEntityCard.js +15 -0
  116. package/dist/esm/resources/MultipleResourceReferenceEditor.js +104 -0
  117. package/dist/esm/resources/MultipleResourceReferenceEditor.spec.js +254 -0
  118. package/dist/esm/resources/SingleResourceReferenceEditor.js +33 -0
  119. package/dist/esm/resources/SingleResourceReferenceEditor.spec.js +118 -0
  120. package/dist/esm/resources/index.js +2 -0
  121. package/dist/esm/resources/testHelpers/resourceEditorHelpers.js +103 -0
  122. package/dist/esm/resources/useResourceLinkActions.js +78 -0
  123. package/dist/esm/types.js +1 -0
  124. package/dist/esm/utils/fromFieldValidations.js +39 -0
  125. package/dist/{__fixtures__ → types/__fixtures__}/FakeSdk.d.ts +8 -8
  126. package/dist/{__fixtures__ → types/__fixtures__}/asset/index.d.ts +6 -6
  127. package/dist/{__fixtures__ → types/__fixtures__}/content-type/index.d.ts +2 -2
  128. package/dist/{__fixtures__ → types/__fixtures__}/entry/index.d.ts +5 -5
  129. package/dist/{__fixtures__ → types/__fixtures__}/fixtures.d.ts +6 -6
  130. package/dist/{__fixtures__ → types/__fixtures__}/locale/index.d.ts +42 -42
  131. package/dist/{__fixtures__ → types/__fixtures__}/space/index.d.ts +2 -2
  132. package/dist/{assets → types/assets}/MultipleMediaEditor.d.ts +10 -10
  133. package/dist/types/assets/SingleMediaEditor.d.ts +10 -0
  134. package/dist/{assets → types/assets}/WrappedAssetCard/AssetCardActions.d.ts +11 -11
  135. package/dist/{assets → types/assets}/WrappedAssetCard/FetchingWrappedAssetCard.d.ts +17 -17
  136. package/dist/{assets → types/assets}/WrappedAssetCard/WrappedAssetCard.d.ts +24 -24
  137. package/dist/{assets → types/assets}/WrappedAssetCard/WrappedAssetLink.d.ts +16 -16
  138. package/dist/{assets → types/assets}/index.d.ts +3 -3
  139. package/dist/{common → types/common}/EntityStore.d.ts +62 -62
  140. package/dist/{common → types/common}/MultipleReferenceEditor.d.ts +25 -25
  141. package/dist/{common → types/common}/ReferenceEditor.d.ts +46 -46
  142. package/dist/{common → types/common}/SingleReferenceEditor.d.ts +24 -24
  143. package/dist/{common → types/common}/SortableLinkList.d.ts +19 -19
  144. package/dist/{common → types/common}/customCardTypes.d.ts +29 -29
  145. package/dist/types/common/useAccessApi.d.ts +16 -0
  146. package/dist/{common → types/common}/useContentTypePermissions.d.ts +17 -17
  147. package/dist/{common → types/common}/useEditorPermissions.d.ts +17 -17
  148. package/dist/types/common/useEditorPermissions.spec.d.ts +1 -0
  149. package/dist/{components → types/components}/AssetThumbnail/AssetThumbnail.d.ts +7 -7
  150. package/dist/{components → types/components}/CreateEntryLinkButton/CreateEntryLinkButton.d.ts +19 -19
  151. package/dist/types/components/CreateEntryLinkButton/CreateEntryLinkButton.spec.d.ts +1 -0
  152. package/dist/{components → types/components}/CreateEntryLinkButton/CreateEntryMenuTrigger.d.ts +31 -31
  153. package/dist/types/components/CreateEntryLinkButton/CreateEntryMenuTrigger.spec.d.ts +1 -0
  154. package/dist/{components → types/components}/CreateEntryLinkButton/useGlobalMouseUp.d.ts +1 -1
  155. package/dist/{components → types/components}/LinkActions/CombinedLinkActions.d.ts +10 -10
  156. package/dist/{components → types/components}/LinkActions/LinkActions.d.ts +26 -26
  157. package/dist/{components → types/components}/LinkActions/LinkEntityActions.d.ts +24 -24
  158. package/dist/types/components/LinkActions/NoLinkPermissionsInfo.d.ts +2 -0
  159. package/dist/{components → types/components}/LinkActions/helpers.d.ts +26 -26
  160. package/dist/{components → types/components}/LinkActions/redesignStyles.d.ts +3 -3
  161. package/dist/{components → types/components}/LinkActions/styles.d.ts +2 -2
  162. package/dist/{components → types/components}/MissingEntityCard/MissingEntityCard.d.ts +8 -8
  163. package/dist/{components → types/components}/MissingEntityCard/styles.d.ts +2 -2
  164. package/dist/{components → types/components}/ScheduledIconWithTooltip/ScheduleTooltip.d.ts +11 -11
  165. package/dist/{components → types/components}/ScheduledIconWithTooltip/ScheduledIconWithTooltip.d.ts +10 -10
  166. package/dist/{components → types/components}/ScheduledIconWithTooltip/formatDateAndTime.d.ts +15 -15
  167. package/dist/types/components/SpaceName/SpaceName.d.ts +6 -0
  168. package/dist/{components → types/components}/index.d.ts +9 -9
  169. package/dist/{entries → types/entries}/MultipleEntryReferenceEditor.d.ts +3 -3
  170. package/dist/{entries → types/entries}/SingleEntryReferenceEditor.d.ts +8 -8
  171. package/dist/{entries → types/entries}/WrappedEntryCard/FetchingWrappedEntryCard.d.ts +18 -18
  172. package/dist/{entries → types/entries}/WrappedEntryCard/WrappedEntryCard.d.ts +35 -35
  173. package/dist/{entries → types/entries}/index.d.ts +3 -3
  174. package/dist/{index.d.ts → types/index.d.ts} +9 -8
  175. package/dist/{resources → types/resources}/Cards/ContentfulEntryCard.d.ts +21 -21
  176. package/dist/{resources → types/resources}/Cards/ResourceCard.d.ts +12 -12
  177. package/dist/{resources → types/resources}/Cards/UnsupportedEntityCard.d.ts +4 -4
  178. package/dist/{resources → types/resources}/MultipleResourceReferenceEditor.d.ts +7 -7
  179. package/dist/types/resources/MultipleResourceReferenceEditor.spec.d.ts +1 -0
  180. package/dist/{resources → types/resources}/SingleResourceReferenceEditor.d.ts +7 -7
  181. package/dist/types/resources/SingleResourceReferenceEditor.spec.d.ts +1 -0
  182. package/dist/{resources → types/resources}/index.d.ts +2 -2
  183. package/dist/{resources → types/resources}/testHelpers/resourceEditorHelpers.d.ts +50 -50
  184. package/dist/{resources → types/resources}/useResourceLinkActions.d.ts +7 -7
  185. package/dist/{types.d.ts → types/types.d.ts} +104 -104
  186. package/dist/{utils → types/utils}/fromFieldValidations.d.ts +21 -21
  187. package/package.json +25 -11
  188. package/CHANGELOG.md +0 -860
  189. package/dist/assets/SingleMediaEditor.d.ts +0 -10
  190. package/dist/common/useAccessApi.d.ts +0 -16
  191. package/dist/components/LinkActions/NoLinkPermissionsInfo.d.ts +0 -2
  192. package/dist/components/SpaceName/SpaceName.d.ts +0 -6
  193. package/dist/field-editor-reference.cjs.development.js +0 -2753
  194. package/dist/field-editor-reference.cjs.development.js.map +0 -1
  195. package/dist/field-editor-reference.cjs.production.min.js +0 -2
  196. package/dist/field-editor-reference.cjs.production.min.js.map +0 -1
  197. package/dist/field-editor-reference.esm.js +0 -2727
  198. package/dist/field-editor-reference.esm.js.map +0 -1
  199. 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,10 @@
1
+ import tokens from '@contentful/f36-tokens';
2
+ import { css } from 'emotion';
3
+ export const container = css({
4
+ display: 'flex',
5
+ width: '100%',
6
+ marginTop: tokens.spacingS
7
+ });
8
+ export const separator = css({
9
+ marginRight: tokens.spacingXl
10
+ });
@@ -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,11 @@
1
+ import { css } from 'emotion';
2
+ export const card = css({
3
+ position: 'relative'
4
+ });
5
+ export const squareCard = css({
6
+ display: 'flex',
7
+ alignItems: 'center',
8
+ width: '135px',
9
+ height: '160px',
10
+ textAlign: 'center'
11
+ });
@@ -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
+ }