@firecms/collection_editor 3.0.0-canary.28 → 3.0.0-canary.280

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 (91) hide show
  1. package/LICENSE +114 -21
  2. package/README.md +165 -1
  3. package/dist/ConfigControllerProvider.d.ts +1 -2
  4. package/dist/index.d.ts +1 -0
  5. package/dist/index.es.js +10792 -4791
  6. package/dist/index.es.js.map +1 -1
  7. package/dist/index.umd.js +11458 -3
  8. package/dist/index.umd.js.map +1 -1
  9. package/dist/types/collection_editor_controller.d.ts +3 -2
  10. package/dist/types/collection_inference.d.ts +4 -1
  11. package/dist/types/config_controller.d.ts +3 -1
  12. package/dist/types/config_permissions.d.ts +2 -2
  13. package/dist/types/persisted_collection.d.ts +1 -1
  14. package/dist/ui/CollectionViewHeaderAction.d.ts +3 -2
  15. package/dist/ui/EditorCollectionActionStart.d.ts +2 -0
  16. package/dist/ui/EditorEntityAction.d.ts +2 -0
  17. package/dist/ui/PropertyAddColumnComponent.d.ts +3 -1
  18. package/dist/ui/collection_editor/CollectionDetailsForm.d.ts +3 -1
  19. package/dist/ui/collection_editor/CollectionEditorDialog.d.ts +3 -2
  20. package/dist/ui/collection_editor/CollectionEditorWelcomeView.d.ts +2 -2
  21. package/dist/ui/collection_editor/CollectionPropertiesEditorForm.d.ts +2 -2
  22. package/dist/ui/collection_editor/EntityActionsEditTab.d.ts +4 -0
  23. package/dist/ui/collection_editor/EntityActionsSelectDialog.d.ts +4 -0
  24. package/dist/ui/collection_editor/LayoutModeSwitch.d.ts +5 -0
  25. package/dist/ui/collection_editor/PropertyEditView.d.ts +8 -0
  26. package/dist/ui/collection_editor/PropertyTree.d.ts +11 -12
  27. package/dist/ui/collection_editor/SubcollectionsEditTab.d.ts +1 -1
  28. package/dist/ui/collection_editor/import/CollectionEditorImportDataPreview.d.ts +1 -1
  29. package/dist/ui/collection_editor/import/CollectionEditorImportMapping.d.ts +8 -1
  30. package/dist/ui/collection_editor/import/clean_import_data.d.ts +1 -1
  31. package/dist/ui/collection_editor/properties/MarkdownPropertyField.d.ts +4 -0
  32. package/dist/ui/collection_editor/properties/ReferencePropertyField.d.ts +2 -1
  33. package/dist/ui/collection_editor/properties/StringPropertyField.d.ts +1 -1
  34. package/dist/useCollectionEditorPlugin.d.ts +8 -11
  35. package/dist/utils/collections.d.ts +6 -0
  36. package/package.json +25 -37
  37. package/src/ConfigControllerProvider.tsx +64 -66
  38. package/src/index.ts +1 -0
  39. package/src/types/collection_editor_controller.tsx +6 -5
  40. package/src/types/collection_inference.ts +4 -1
  41. package/src/types/config_controller.tsx +4 -1
  42. package/src/types/config_permissions.ts +1 -1
  43. package/src/types/persisted_collection.ts +2 -3
  44. package/src/ui/CollectionViewHeaderAction.tsx +10 -5
  45. package/src/ui/EditorCollectionAction.tsx +12 -70
  46. package/src/ui/EditorCollectionActionStart.tsx +87 -0
  47. package/src/ui/EditorEntityAction.tsx +51 -0
  48. package/src/ui/HomePageEditorCollectionAction.tsx +21 -14
  49. package/src/ui/NewCollectionButton.tsx +1 -1
  50. package/src/ui/NewCollectionCard.tsx +3 -3
  51. package/src/ui/PropertyAddColumnComponent.tsx +11 -6
  52. package/src/ui/collection_editor/CollectionDetailsForm.tsx +157 -50
  53. package/src/ui/collection_editor/CollectionEditorDialog.tsx +119 -39
  54. package/src/ui/collection_editor/CollectionEditorWelcomeView.tsx +24 -33
  55. package/src/ui/collection_editor/CollectionPropertiesEditorForm.tsx +46 -41
  56. package/src/ui/collection_editor/EntityActionsEditTab.tsx +163 -0
  57. package/src/ui/collection_editor/EntityActionsSelectDialog.tsx +41 -0
  58. package/src/ui/collection_editor/EntityCustomViewsSelectDialog.tsx +11 -7
  59. package/src/ui/collection_editor/EnumForm.tsx +11 -7
  60. package/src/ui/collection_editor/GetCodeDialog.tsx +60 -28
  61. package/src/ui/collection_editor/LayoutModeSwitch.tsx +54 -0
  62. package/src/ui/collection_editor/PropertyEditView.tsx +266 -79
  63. package/src/ui/collection_editor/PropertyFieldPreview.tsx +8 -10
  64. package/src/ui/collection_editor/PropertyTree.tsx +184 -138
  65. package/src/ui/collection_editor/SubcollectionsEditTab.tsx +26 -19
  66. package/src/ui/collection_editor/UnsavedChangesDialog.tsx +9 -7
  67. package/src/ui/collection_editor/import/CollectionEditorImportDataPreview.tsx +41 -9
  68. package/src/ui/collection_editor/import/CollectionEditorImportMapping.tsx +43 -10
  69. package/src/ui/collection_editor/import/clean_import_data.ts +1 -1
  70. package/src/ui/collection_editor/properties/BlockPropertyField.tsx +32 -20
  71. package/src/ui/collection_editor/properties/DateTimePropertyField.tsx +54 -47
  72. package/src/ui/collection_editor/properties/EnumPropertyField.tsx +3 -1
  73. package/src/ui/collection_editor/properties/MapPropertyField.tsx +8 -7
  74. package/src/ui/collection_editor/properties/MarkdownPropertyField.tsx +139 -0
  75. package/src/ui/collection_editor/properties/ReferencePropertyField.tsx +7 -3
  76. package/src/ui/collection_editor/properties/RepeatPropertyField.tsx +0 -1
  77. package/src/ui/collection_editor/properties/StoragePropertyField.tsx +34 -19
  78. package/src/ui/collection_editor/properties/StringPropertyField.tsx +4 -9
  79. package/src/ui/collection_editor/properties/UrlPropertyField.tsx +1 -0
  80. package/src/ui/collection_editor/properties/advanced/AdvancedPropertyValidation.tsx +2 -0
  81. package/src/ui/collection_editor/properties/validation/ValidationPanel.tsx +2 -2
  82. package/src/ui/collection_editor/templates/pages_template.ts +1 -6
  83. package/src/ui/collection_editor/utils/strings.ts +13 -6
  84. package/src/ui/collection_editor/utils/supported_fields.tsx +2 -0
  85. package/src/ui/collection_editor/utils/update_property_for_widget.ts +37 -6
  86. package/src/useCollectionEditorPlugin.tsx +38 -32
  87. package/src/utils/collections.ts +46 -0
  88. package/dist/ui/RootCollectionSuggestions.d.ts +0 -3
  89. package/dist/ui/collection_editor/PropertySelectItem.d.ts +0 -8
  90. package/src/ui/RootCollectionSuggestions.tsx +0 -63
  91. package/src/ui/collection_editor/PropertySelectItem.tsx +0 -32
@@ -3,17 +3,16 @@ import {
3
3
  getInferenceType,
4
4
  ImportConfig,
5
5
  ImportNewPropertyFieldPreview
6
- } from "@firecms/data_import_export";
6
+ } from "@firecms/data_import";
7
7
  import { getIn, useFormex } from "@firecms/formex";
8
8
 
9
- import { PropertyConfigBadge, getFieldConfig, getFieldId, Properties, Property, PropertyConfig, } from "@firecms/core";
10
- import { Container, Select, Tooltip, Typography } from "@firecms/ui";
9
+ import { getFieldConfig, getFieldId, Properties, Property, PropertyConfig, PropertyConfigBadge, } from "@firecms/core";
10
+ import { cls, Container, Select, SelectItem, Tooltip, Typography } from "@firecms/ui";
11
11
  import React, { useState } from "react";
12
12
  import { OnPropertyChangedParams, PropertyFormDialog, PropertyWithId } from "../PropertyEditView";
13
13
  import { getFullId, idToPropertiesPath, namespaceToPropertiesOrderPath } from "../util";
14
14
  import { PersistedCollection } from "../../../types/persisted_collection";
15
15
  import { updatePropertyFromWidget } from "../utils/update_property_for_widget";
16
- import { PropertySelectItem } from "../PropertySelectItem";
17
16
  import { supportedFields } from "../utils/supported_fields";
18
17
  import { buildPropertyFromData } from "@firecms/schema_inference";
19
18
 
@@ -143,18 +142,20 @@ export function CollectionEditorImportMapping({
143
142
  <div className={"overflow-auto my-auto"}>
144
143
  <Container maxWidth={"6xl"} className={"flex flex-col gap-4 p-8 m-auto"}>
145
144
 
146
- <Typography variant="h6" className={"mt-4"}>Data property mapping</Typography>
145
+ <Typography variant="h6" className={"my-4 ml-3.5"}>Data property mapping</Typography>
147
146
 
148
- <DataNewPropertiesMapping headersMapping={importConfig.headersMapping}
149
- idColumn={importConfig.idColumn}
150
- originProperties={importConfig.originProperties}
147
+ <DataNewPropertiesMapping importConfig={importConfig}
151
148
  destinationProperties={values.properties as Properties}
152
- onIdPropertyChanged={(value) => importConfig.setIdColumn(value ?? undefined)}
153
149
  buildPropertyView={({
154
150
  property,
155
151
  propertyKey,
156
- importKey
152
+ importKey,
153
+ isIdColumn
157
154
  }) => {
155
+ if (isIdColumn) {
156
+ return <Typography> This column will be used as ID</Typography>
157
+ }
158
+
158
159
  return <ImportNewPropertyFieldPreview
159
160
  property={property}
160
161
  propertyKey={propertyKey}
@@ -234,9 +235,11 @@ function PropertySelect({
234
235
  open={selectOpen}
235
236
  onOpenChange={setSelectOpen}
236
237
  invisible={true}
238
+ size={"large"}
237
239
  className={"w-full"}
238
240
  disabled={disabled}
239
241
  error={!widget}
242
+ fullWidth={true}
240
243
  value={fieldId ?? ""}
241
244
  placeholder={"Select a property widget"}
242
245
  position={"item-aligned"}
@@ -266,3 +269,33 @@ function PropertySelect({
266
269
  </Select>
267
270
  </Tooltip>;
268
271
  }
272
+
273
+ export interface PropertySelectItemProps {
274
+ value: string;
275
+ optionDisabled: boolean;
276
+ propertyConfig: PropertyConfig;
277
+ existing: boolean;
278
+ }
279
+
280
+ export function PropertySelectItem({ value, optionDisabled, propertyConfig, existing }: PropertySelectItemProps) {
281
+ return <SelectItem value={value}
282
+ disabled={optionDisabled}
283
+ className={"flex flex-row items-center"}>
284
+ <div
285
+ className={cls(
286
+ "flex flex-row items-center text-base min-h-[48px]",
287
+ optionDisabled ? "w-full" : "")}>
288
+ <div className={"mr-8"}>
289
+ <PropertyConfigBadge propertyConfig={propertyConfig}/>
290
+ </div>
291
+ <div>
292
+ <div>{propertyConfig.name}</div>
293
+ <Typography variant={"caption"}
294
+ color={"disabled"}
295
+ className={"max-w-sm"}>
296
+ {existing && optionDisabled ? "You can only switch to widgets that use the same data type" : propertyConfig.description}
297
+ </Typography>
298
+ </div>
299
+ </div>
300
+ </SelectItem>
301
+ }
@@ -1,5 +1,5 @@
1
1
  import { Properties, slugify } from "@firecms/core";
2
- import { ImportConfig } from "@firecms/data_import_export";
2
+ import { ImportConfig } from "@firecms/data_import";
3
3
 
4
4
  export function cleanPropertiesFromImport(properties: Properties, parentSlug = ""): {
5
5
  headersMapping: ImportConfig["headersMapping"],
@@ -1,12 +1,24 @@
1
- import React, { useCallback, useState } from "react";
1
+ import React, { useState } from "react";
2
2
  import { AddIcon, Button, Paper, Typography } from "@firecms/ui";
3
3
  import { getIn, useFormex } from "@firecms/formex";
4
- import { PropertyFormDialog } from "../PropertyEditView";
5
- import { getFullId, idToPropertiesPath, namespaceToPropertiesOrderPath } from "../util";
4
+ import { OnPropertyChangedParams, PropertyFormDialog } from "../PropertyEditView";
5
+ import {
6
+ getFullId,
7
+ getFullIdPath,
8
+ idToPropertiesPath,
9
+ namespaceToPropertiesOrderPath,
10
+ namespaceToPropertiesPath
11
+ } from "../util";
6
12
  import { PropertyTree } from "../PropertyTree";
7
- import { ArrayProperty, Property, PropertyConfig } from "@firecms/core";
8
-
9
- export function BlockPropertyField({ disabled, getData, allowDataInference, propertyConfigs, collectionEditable }: {
13
+ import { ArrayProperty, PropertyConfig } from "@firecms/core";
14
+
15
+ export function BlockPropertyField({
16
+ disabled,
17
+ getData,
18
+ allowDataInference,
19
+ propertyConfigs,
20
+ collectionEditable
21
+ }: {
10
22
  disabled: boolean;
11
23
  getData?: () => Promise<object[]>;
12
24
  allowDataInference: boolean;
@@ -25,38 +37,37 @@ export function BlockPropertyField({ disabled, getData, allowDataInference, prop
25
37
 
26
38
  const onPropertyChanged = ({
27
39
  id,
40
+ namespace,
28
41
  property
29
- }: { id?: string, property: Property }) => {
42
+ }: OnPropertyChangedParams) => {
30
43
  if (!id)
31
44
  throw Error();
32
45
 
33
- setFieldValue("oneOf.properties", {
34
- ...(values.oneOf?.properties ?? {}),
35
- [id]: property
36
- }, false);
46
+ setFieldValue("oneOf." + getFullIdPath(id, namespace), property, false);
47
+
37
48
  const currentPropertiesOrder = values.oneOf?.propertiesOrder ?? Object.keys(values.oneOf?.properties ?? {});
38
49
  const newPropertiesOrder = currentPropertiesOrder.includes(id) ? currentPropertiesOrder : [...currentPropertiesOrder, id];
39
- setFieldValue("oneOf.propertiesOrder", newPropertiesOrder, false);
50
+ setFieldValue("oneOf." + namespaceToPropertiesOrderPath(namespace), newPropertiesOrder, false);
40
51
  setPropertyDialogOpen(false);
41
52
  };
42
53
 
43
54
  const selectedPropertyFullId = selectedPropertyKey ? getFullId(selectedPropertyKey, selectedPropertyNamespace) : undefined;
44
55
  const selectedProperty = selectedPropertyFullId ? getIn(values.oneOf?.properties, selectedPropertyFullId.replaceAll(".", ".properties.")) : undefined;
45
56
 
46
- const deleteProperty = useCallback((propertyKey?: string, namespace?: string) => {
57
+ const deleteProperty = (propertyKey?: string, namespace?: string) => {
47
58
  const fullId = propertyKey ? getFullId(propertyKey, namespace) : undefined;
48
59
  if (!fullId)
49
60
  throw Error("collection editor miss config");
50
61
 
51
62
  setFieldValue(`oneOf.${idToPropertiesPath(fullId)}`, undefined, false);
52
63
  const propertiesOrderPath = `oneOf.${namespaceToPropertiesOrderPath(namespace)}`;
53
- const currentPropertiesOrder: string[] = getIn(values, propertiesOrderPath);
64
+ const currentPropertiesOrder: string[] = getIn(values, propertiesOrderPath) ?? Object.keys(getIn(values, namespaceToPropertiesPath(namespace)));
54
65
  setFieldValue(propertiesOrderPath, currentPropertiesOrder.filter((p) => p !== propertyKey), false);
55
66
 
56
67
  setPropertyDialogOpen(false);
57
68
  setSelectedPropertyKey(undefined);
58
69
  setSelectedPropertyNamespace(undefined);
59
- }, [setFieldValue, values]);
70
+ };
60
71
 
61
72
  const addChildButton = <Button
62
73
  autoFocus
@@ -68,16 +79,17 @@ export function BlockPropertyField({ disabled, getData, allowDataInference, prop
68
79
  Add property to {values.name ?? "this block"}
69
80
  </Button>;
70
81
 
71
- const onPropertyMove = useCallback((propertiesOrder: string[], namespace?: string) => {
82
+ const onPropertyMove = (propertiesOrder: string[], namespace?: string) => {
72
83
  setFieldValue(`oneOf.${namespaceToPropertiesOrderPath(namespace)}`, propertiesOrder, false);
73
- }, []);
84
+ };
74
85
 
75
86
  return (
76
87
  <>
77
88
  <div className={"col-span-12"}>
78
89
  <div className={"flex justify-between items-end mt-8 mb-4"}>
79
- <Typography variant={"subtitle2"}>Properties in this
80
- block</Typography>
90
+ <Typography variant={"subtitle2"}>
91
+ Properties in this block
92
+ </Typography>
81
93
  {addChildButton}
82
94
  </div>
83
95
  <Paper className="p-2 pl-8">
@@ -98,7 +110,7 @@ export function BlockPropertyField({ disabled, getData, allowDataInference, prop
98
110
  ? undefined
99
111
  : onPropertyMove}/>
100
112
 
101
- {!disabled && !values.oneOf?.propertiesOrder?.length &&
113
+ {!disabled && (values.oneOf?.propertiesOrder?.length === 0)&&
102
114
  <div className="h-full flex items-center justify-center p-4">
103
115
  Add the first property to this block
104
116
  </div>}
@@ -26,53 +26,60 @@ export function DateTimePropertyField({ disabled }: {
26
26
 
27
27
  return (
28
28
  <>
29
- <div className={"flex flex-col col-span-12"}>
30
- <Select name={modePath}
31
- value={modeValue ?? "date"}
32
- error={Boolean(modeError)}
33
- onValueChange={(v) => setFieldValue(modePath, v)}
34
- label={"Mode"}
35
- renderValue={(v) => {
36
- switch (v) {
37
- case "date_time":
38
- return "Date/Time";
39
- case "date":
40
- return "Date";
41
- default:
42
- return "";
43
- }
44
- }}
45
- disabled={disabled}>
46
- <SelectItem value={"date_time"}> Date/Time </SelectItem>
47
- <SelectItem value={"date"}> Date </SelectItem>
48
- </Select>
49
- <FieldCaption error={Boolean(modeError)}>
50
- {modeError}
51
- </FieldCaption>
52
-
53
- <Select name={autoValuePath}
54
- disabled={disabled}
55
- value={autoValueValue ?? ""}
56
- onValueChange={(v) => setFieldValue(autoValuePath, v === "none" ? null : v)}
57
- renderValue={(v) => {
58
- switch (v) {
59
- case "on_create":
60
- return "On create";
61
- case "on_update":
62
- return "On any update";
63
- default:
64
- return "None";
65
- }
66
- }}
67
- error={Boolean(autoValueError)}
68
- label={"Automatic value"}>
69
- <SelectItem value={"none"}> None </SelectItem>
70
- <SelectItem value={"on_create"}> On create </SelectItem>
71
- <SelectItem value={"on_update"}> On any update </SelectItem>
72
- </Select>
73
- <FieldCaption error={Boolean(autoValueError)}>
74
- {autoValueError ?? "Update this field automatically when creating or updating the entity"}
75
- </FieldCaption>
29
+ <div className={"flex flex-col col-span-12 gap-2"}>
30
+ <div>
31
+ <Select name={modePath}
32
+ value={modeValue ?? "date"}
33
+ error={Boolean(modeError)}
34
+ size={"large"}
35
+ onValueChange={(v) => setFieldValue(modePath, v)}
36
+ label={"Mode"}
37
+ fullWidth={true}
38
+ renderValue={(v) => {
39
+ switch (v) {
40
+ case "date_time":
41
+ return "Date/Time";
42
+ case "date":
43
+ return "Date";
44
+ default:
45
+ return "";
46
+ }
47
+ }}
48
+ disabled={disabled}>
49
+ <SelectItem value={"date_time"}> Date/Time </SelectItem>
50
+ <SelectItem value={"date"}> Date </SelectItem>
51
+ </Select>
52
+ <FieldCaption error={Boolean(modeError)}>
53
+ {modeError}
54
+ </FieldCaption>
55
+ </div>
56
+ <div>
57
+ <Select name={autoValuePath}
58
+ disabled={disabled}
59
+ size={"large"}
60
+ fullWidth={true}
61
+ value={autoValueValue ?? ""}
62
+ onValueChange={(v) => setFieldValue(autoValuePath, v === "none" ? null : v)}
63
+ renderValue={(v) => {
64
+ switch (v) {
65
+ case "on_create":
66
+ return "On create";
67
+ case "on_update":
68
+ return "On any update";
69
+ default:
70
+ return "None";
71
+ }
72
+ }}
73
+ error={Boolean(autoValueError)}
74
+ label={"Automatic value"}>
75
+ <SelectItem value={"none"}> None </SelectItem>
76
+ <SelectItem value={"on_create"}> On create </SelectItem>
77
+ <SelectItem value={"on_update"}> On any update </SelectItem>
78
+ </Select>
79
+ <FieldCaption error={Boolean(autoValueError)}>
80
+ {autoValueError ?? "Update this field automatically when creating or updating the entity"}
81
+ </FieldCaption>
82
+ </div>
76
83
 
77
84
  </div>
78
85
 
@@ -71,7 +71,7 @@ export function EnumPropertyField({
71
71
  }}
72
72
  getData={getData
73
73
  ? () => getData()
74
- .then(res => res.map(d => values.id && getIn(d, values.id)).filter(Boolean))
74
+ .then(res => res.map(entry => values.id && getIn(entry, values.id)).filter(Boolean))
75
75
  : undefined}
76
76
  onValuesChanged={onValuesChanged}/>
77
77
  </div>
@@ -93,9 +93,11 @@ export function EnumPropertyField({
93
93
  <Select
94
94
  disabled={disabled}
95
95
  position={"item-aligned"}
96
+ fullWidth={true}
96
97
  onValueChange={(value: string) => {
97
98
  setFieldValue("defaultValue", value);
98
99
  }}
100
+ size={"large"}
99
101
  label={"Default value"}
100
102
  value={defaultValue ?? ""}>
101
103
  {enumValues
@@ -1,4 +1,4 @@
1
- import React, { useCallback, useState } from "react";
1
+ import React, { useState } from "react";
2
2
  import { FieldCaption, MapProperty, Property, PropertyConfig, } from "@firecms/core";
3
3
  import { AddIcon, BooleanSwitchWithLabel, Button, Paper, Typography } from "@firecms/ui";
4
4
  import { PropertyFormDialog } from "../PropertyEditView";
@@ -42,7 +42,7 @@ export function MapPropertyField({ disabled, getData, allowDataInference, proper
42
42
  setPropertyDialogOpen(false);
43
43
  };
44
44
 
45
- const deleteProperty = useCallback((propertyKey?: string, namespace?: string) => {
45
+ const deleteProperty = (propertyKey?: string, namespace?: string) => {
46
46
  const fullId = propertyKey ? getFullId(propertyKey, namespace) : undefined;
47
47
  if (!fullId)
48
48
  throw Error("collection editor miss config");
@@ -58,16 +58,16 @@ export function MapPropertyField({ disabled, getData, allowDataInference, proper
58
58
  setPropertyDialogOpen(false);
59
59
  setSelectedPropertyKey(undefined);
60
60
  setSelectedPropertyNamespace(undefined);
61
- }, [setFieldValue, values]);
61
+ };
62
62
 
63
63
  const selectedPropertyFullId = selectedPropertyKey ? getFullId(selectedPropertyKey, selectedPropertyNamespace) : undefined;
64
64
  const selectedProperty = selectedPropertyFullId ? getIn(values.properties, selectedPropertyFullId.replaceAll(".", ".properties.")) : undefined;
65
65
 
66
66
  const empty = !propertiesOrder || propertiesOrder.length < 1;
67
67
 
68
- const onPropertyMove = useCallback((propertiesOrder: string[], namespace?: string) => {
68
+ const onPropertyMove = (propertiesOrder: string[], namespace?: string) => {
69
69
  setFieldValue(namespaceToPropertiesOrderPath(namespace), propertiesOrder, false);
70
- }, []);
70
+ };
71
71
 
72
72
  return (
73
73
  <>
@@ -107,13 +107,14 @@ export function MapPropertyField({ disabled, getData, allowDataInference, proper
107
107
  <div className={"col-span-12"}>
108
108
  <BooleanSwitchWithLabel
109
109
  position={"start"}
110
- size={"small"}
110
+ size={"medium"}
111
111
  label="Spread children as columns"
112
112
  onValueChange={(v) => setFieldValue("spreadChildren", v)}
113
113
  value={values.spreadChildren ?? false}
114
114
  />
115
115
  <FieldCaption>
116
- Set this flag to true if you want to display the children of this group as individual columns.
116
+ Set this flag to true if you want to display the children of this group as individual columns. This
117
+ will only work for top level groups.
117
118
  </FieldCaption>
118
119
  </div>
119
120
 
@@ -0,0 +1,139 @@
1
+ import React from "react";
2
+ import { StringPropertyValidation } from "./validation/StringPropertyValidation";
3
+ import { ValidationPanel } from "./validation/ValidationPanel";
4
+ import { Field, getIn, useFormex } from "@firecms/formex";
5
+
6
+ import { CloudUploadIcon, DebouncedTextField, ExpandablePanel, TextField, Typography } from "@firecms/ui";
7
+
8
+ export function MarkdownPropertyField({
9
+ disabled,
10
+ showErrors
11
+ }: {
12
+ disabled: boolean;
13
+ showErrors: boolean;
14
+ }) {
15
+
16
+ const {
17
+ values,
18
+ setFieldValue
19
+ } = useFormex();
20
+
21
+ const baseStoragePath = "storage";
22
+
23
+ const metadata = `${baseStoragePath}.metadata`;
24
+ const fileName = `${baseStoragePath}.fileName`;
25
+ const maxSize = `${baseStoragePath}.maxSize`;
26
+ const storagePath = `${baseStoragePath}.storagePath`;
27
+
28
+ const fileNameValue = getIn(values, fileName) ?? "{rand}_{file}";
29
+ const storagePathValue = getIn(values, storagePath) ?? "/";
30
+ const maxSizeValue = getIn(values, maxSize);
31
+
32
+ const hasFilenameCallback = typeof fileNameValue === "function";
33
+ const hasStoragePathCallback = typeof storagePathValue === "function";
34
+
35
+ return (
36
+ <>
37
+ <div className={"col-span-12"}>
38
+
39
+ <ValidationPanel>
40
+
41
+ <StringPropertyValidation disabled={disabled}
42
+ length={true}
43
+ lowercase={true}
44
+ max={true}
45
+ min={true}
46
+ trim={true}
47
+ uppercase={true}
48
+ showErrors={showErrors}/>
49
+
50
+ </ValidationPanel>
51
+
52
+ </div>
53
+
54
+ <div className={"col-span-12"}>
55
+ <ExpandablePanel
56
+ title={
57
+ <div className="flex flex-row text-surface-500">
58
+ <CloudUploadIcon/>
59
+ <Typography variant={"subtitle2"}
60
+ className="ml-2">
61
+ File upload config
62
+ </Typography>
63
+ </div>
64
+ }>
65
+
66
+ <div className={"grid grid-cols-12 gap-2 p-4"}>
67
+
68
+
69
+ <div className={"col-span-12"}>
70
+ <Field name={fileName}
71
+ as={DebouncedTextField}
72
+ label={"File name"}
73
+ size={"small"}
74
+ disabled={hasFilenameCallback || disabled}
75
+ value={hasFilenameCallback ? "-" : fileNameValue}
76
+ />
77
+ </div>
78
+ <div className={"col-span-12"}>
79
+ <Field name={storagePath}
80
+ as={DebouncedTextField}
81
+ label={"Storage path"}
82
+ disabled={hasStoragePathCallback || disabled}
83
+ size={"small"}
84
+ value={hasStoragePathCallback ? "-" : storagePathValue}
85
+ />
86
+ <Typography variant={"caption"} className={"ml-3.5 mt-1 mb-2"}>
87
+ <p>You can use the following placeholders in
88
+ the file name
89
+ and storage path values:</p>
90
+ <ul>
91
+ <li>{"{file} - Full name of the uploaded file"}</li>
92
+ <li>{"{file.name} - Name of the uploaded file without extension"}</li>
93
+ <li>{"{file.ext} - Extension of the uploaded file"}</li>
94
+ <li>{"{entityId} - ID of the entity"}</li>
95
+ <li>{"{propertyKey} - ID of this field"}</li>
96
+ <li>{"{path} - Path of this entity"}</li>
97
+ <li>{"{rand} - Random value used to avoid name collisions"}</li>
98
+ </ul>
99
+ </Typography>
100
+
101
+ <Typography variant={"caption"} className={"ml-3.5 mt-1 mb-2"}>
102
+ When using Markdown, the URL of the uploaded files are always saved in the text value
103
+ (not
104
+ the path).
105
+ </Typography>
106
+ </div>
107
+
108
+ <div className={"col-span-12"}>
109
+ <DebouncedTextField name={maxSize}
110
+ type={"number"}
111
+ label={"Max size (in bytes)"}
112
+ size={"small"}
113
+ value={maxSizeValue !== undefined && maxSizeValue !== null ? maxSizeValue.toString() : ""}
114
+ onChange={(e) => {
115
+ const value = e.target.value;
116
+ if (value === "") setFieldValue(maxSize, undefined);
117
+ else setFieldValue(maxSize, parseInt(value));
118
+ }}
119
+ />
120
+ </div>
121
+
122
+ </div>
123
+ </ExpandablePanel>
124
+ </div>
125
+
126
+ <div className={"col-span-12"}>
127
+
128
+ <TextField name={"defaultValue"}
129
+ disabled={disabled}
130
+ onChange={(e: any) => {
131
+ setFieldValue("defaultValue", e.target.value === "" ? undefined : e.target.value);
132
+ }}
133
+ label={"Default value"}
134
+ value={getIn(values, "defaultValue") ?? ""}/>
135
+
136
+ </div>
137
+ </>
138
+ );
139
+ }
@@ -7,12 +7,14 @@ export function ReferencePropertyField({
7
7
  existing,
8
8
  multiple,
9
9
  disabled,
10
- showErrors
10
+ showErrors,
11
+ asString
11
12
  }: {
12
13
  existing: boolean,
13
14
  multiple: boolean,
14
15
  disabled: boolean,
15
- showErrors: boolean
16
+ showErrors: boolean,
17
+ asString?: boolean
16
18
  }) {
17
19
 
18
20
  const {
@@ -28,7 +30,7 @@ export function ReferencePropertyField({
28
30
  <CircularProgress/>
29
31
  </div>;
30
32
 
31
- const pathPath = multiple ? "of.path" : "path";
33
+ const pathPath = asString ? "reference.path" : (multiple ? "of.path" : "path") ;
32
34
  const pathValue: string | undefined = getIn(values, pathPath);
33
35
  const pathError: string | undefined = showErrors && getIn(errors, pathPath);
34
36
 
@@ -89,6 +91,8 @@ export function CollectionsSelect({
89
91
  value={value ?? ""}
90
92
  position={"item-aligned"}
91
93
  name={pathPath}
94
+ size={"large"}
95
+ fullWidth={true}
92
96
  onChange={handleChange}
93
97
  label={"Target collection"}
94
98
  renderValue={(selected) => {
@@ -39,7 +39,6 @@ export function RepeatPropertyField({
39
39
 
40
40
  const onPropertyChanged = ({ id, property, namespace }:
41
41
  { id?: string, property: Property, namespace?: string }) => {
42
- console.log("onPropertyChanged", id, property, namespace);
43
42
  setFieldValue("of", property);
44
43
  };
45
44