@arpproject/recrate 0.1.2

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 (286) hide show
  1. package/.eslintrc.json +37 -0
  2. package/.storybook/main.ts +40 -0
  3. package/.storybook/preview.tsx +46 -0
  4. package/LICENCE +18 -0
  5. package/README.md +83 -0
  6. package/babel.config.json +5 -0
  7. package/dist/app/App.d.ts +2 -0
  8. package/dist/app/EmbeddedComponent.d.ts +3 -0
  9. package/dist/app/index.d.ts +0 -0
  10. package/dist/app/lookup.d.ts +82 -0
  11. package/dist/crate-builder/CrateManager/contexts.d.ts +6 -0
  12. package/dist/crate-builder/CrateManager/crate-manager-benchmarking.spec.d.ts +1 -0
  13. package/dist/crate-builder/CrateManager/crate-manager-loading-exporting.spec.d.ts +1 -0
  14. package/dist/crate-builder/CrateManager/crate-manager-operations.spec.d.ts +0 -0
  15. package/dist/crate-builder/CrateManager/crate-manager.d.ts +641 -0
  16. package/dist/crate-builder/CrateManager/crate-manager.spec.d.ts +1 -0
  17. package/dist/crate-builder/CrateManager/lib.d.ts +26 -0
  18. package/dist/crate-builder/CrateManager/lib.spec.d.ts +1 -0
  19. package/dist/crate-builder/CrateManager/profile-manager.d.ts +143 -0
  20. package/dist/crate-builder/CrateManager/profile-manager.spec.d.ts +1 -0
  21. package/dist/crate-builder/CrateManager/schema-type-definitions.json.d.ts +35125 -0
  22. package/dist/crate-builder/CrateManager/validate-identifier.d.ts +10 -0
  23. package/dist/crate-builder/CrateManager/validate-identifier.spec.d.ts +1 -0
  24. package/dist/crate-builder/RenderEntity/Add.d.ts +26 -0
  25. package/dist/crate-builder/RenderEntity/AddControl.d.ts +14 -0
  26. package/dist/crate-builder/RenderEntity/AddControl.stories.d.ts +29 -0
  27. package/dist/crate-builder/RenderEntity/AutoComplete.d.ts +12 -0
  28. package/dist/crate-builder/RenderEntity/BulkAdd.d.ts +10 -0
  29. package/dist/crate-builder/RenderEntity/DeleteProperty.d.ts +8 -0
  30. package/dist/crate-builder/RenderEntity/DialogAddProperty.d.ts +9 -0
  31. package/dist/crate-builder/RenderEntity/DialogBrowseEntities.d.ts +8 -0
  32. package/dist/crate-builder/RenderEntity/DialogEditContext.d.ts +8 -0
  33. package/dist/crate-builder/RenderEntity/DialogPreviewCrate.d.ts +7 -0
  34. package/dist/crate-builder/RenderEntity/DialogSaveCrateAsTemplate.d.ts +13 -0
  35. package/dist/crate-builder/RenderEntity/DialogSaveEntityTemplate.d.ts +14 -0
  36. package/dist/crate-builder/RenderEntity/DisplayPropertyName.d.ts +8 -0
  37. package/dist/crate-builder/RenderEntity/DisplayPropertyName.stories.d.ts +6 -0
  38. package/dist/crate-builder/RenderEntity/EntityId.d.ts +13 -0
  39. package/dist/crate-builder/RenderEntity/EntityName.d.ts +13 -0
  40. package/dist/crate-builder/RenderEntity/EntityProperty.d.ts +19 -0
  41. package/dist/crate-builder/RenderEntity/EntityPropertyInstance.d.ts +25 -0
  42. package/dist/crate-builder/RenderEntity/EntityType.d.ts +12 -0
  43. package/dist/crate-builder/RenderEntity/ItemLink.d.ts +14 -0
  44. package/dist/crate-builder/RenderEntity/PaginateLinkedEntities.d.ts +23 -0
  45. package/dist/crate-builder/RenderEntity/PaginateLinkedEntities.stories.d.ts +7 -0
  46. package/dist/crate-builder/RenderEntity/PropertyHelp.d.ts +6 -0
  47. package/dist/crate-builder/RenderEntity/RenderControls.d.ts +22 -0
  48. package/dist/crate-builder/RenderEntity/RenderLinkedItem.d.ts +20 -0
  49. package/dist/crate-builder/RenderEntity/RenderPropertyHelp.d.ts +6 -0
  50. package/dist/crate-builder/RenderEntity/RenderReverseConnections.d.ts +16 -0
  51. package/dist/crate-builder/RenderEntity/RenderTypes.d.ts +6 -0
  52. package/dist/crate-builder/RenderEntity/Shell2.d.ts +14 -0
  53. package/dist/crate-builder/RenderEntity/UnlinkEntity.d.ts +14 -0
  54. package/dist/crate-builder/RenderEntity/auto-complete.lib.d.ts +25 -0
  55. package/dist/crate-builder/RenderEntity/keys.d.ts +4 -0
  56. package/dist/crate-builder/RenderEntity/layout.d.ts +14 -0
  57. package/dist/crate-builder/RenderEntity/layout.spec.d.ts +1 -0
  58. package/dist/crate-builder/Shell.d.ts +40 -0
  59. package/dist/crate-builder/editor-state.d.ts +72 -0
  60. package/dist/crate-builder/helpers.d.ts +9 -0
  61. package/dist/crate-builder/i18n.d.ts +2 -0
  62. package/dist/crate-builder/lib/validate-iri.d.ts +6 -0
  63. package/dist/crate-builder/locales/en.d.ts +79 -0
  64. package/dist/crate-builder/locales/hu.d.ts +79 -0
  65. package/dist/crate-builder/primitives/Boolean.d.ts +11 -0
  66. package/dist/crate-builder/primitives/Boolean.stories.d.ts +7 -0
  67. package/dist/crate-builder/primitives/Date.d.ts +11 -0
  68. package/dist/crate-builder/primitives/Date.stories.d.ts +5 -0
  69. package/dist/crate-builder/primitives/DateTime.d.ts +11 -0
  70. package/dist/crate-builder/primitives/DateTime.stories.d.ts +5 -0
  71. package/dist/crate-builder/primitives/Geo.d.ts +12 -0
  72. package/dist/crate-builder/primitives/Geo.stories.d.ts +26 -0
  73. package/dist/crate-builder/primitives/Map.SelectArea.d.ts +9 -0
  74. package/dist/crate-builder/primitives/Map.d.ts +15 -0
  75. package/dist/crate-builder/primitives/Map.stories.d.ts +6 -0
  76. package/dist/crate-builder/primitives/Number.d.ts +16 -0
  77. package/dist/crate-builder/primitives/Number.stories.d.ts +10 -0
  78. package/dist/crate-builder/primitives/Select.d.ts +15 -0
  79. package/dist/crate-builder/primitives/Select.stories.d.ts +9 -0
  80. package/dist/crate-builder/primitives/SelectObject.d.ts +13 -0
  81. package/dist/crate-builder/primitives/SelectObject.stories.d.ts +5 -0
  82. package/dist/crate-builder/primitives/SelectUrl.d.ts +14 -0
  83. package/dist/crate-builder/primitives/SelectUrl.stories.d.ts +7 -0
  84. package/dist/crate-builder/primitives/Text.d.ts +19 -0
  85. package/dist/crate-builder/primitives/Text.stories.d.ts +13 -0
  86. package/dist/crate-builder/primitives/Time.d.ts +11 -0
  87. package/dist/crate-builder/primitives/Time.stories.d.ts +7 -0
  88. package/dist/crate-builder/primitives/Url.d.ts +11 -0
  89. package/dist/crate-builder/primitives/Url.stories.d.ts +7 -0
  90. package/dist/crate-builder/primitives/Value.d.ts +7 -0
  91. package/dist/crate-builder/primitives/Value.stories.d.ts +7 -0
  92. package/dist/crate-builder/primitives/date-libs.d.ts +1 -0
  93. package/dist/crate-builder/property-definitions.d.ts +78 -0
  94. package/dist/crate-builder/store.d.ts +8 -0
  95. package/dist/favicon.ico +0 -0
  96. package/dist/index.d.ts +3 -0
  97. package/dist/index.html +43 -0
  98. package/dist/logo192.png +0 -0
  99. package/dist/logo512.png +0 -0
  100. package/dist/manifest.json +25 -0
  101. package/dist/marker-icon.png +0 -0
  102. package/dist/marker-shadow.png +0 -0
  103. package/dist/recrate.es.js +158699 -0
  104. package/dist/recrate.umd.js +158717 -0
  105. package/dist/robots.txt +3 -0
  106. package/dist/style.css +1721 -0
  107. package/dist/types.d.ts +42 -0
  108. package/docker-compose.yml +30 -0
  109. package/docs/.nojekyll +1 -0
  110. package/docs/assets/highlight.css +99 -0
  111. package/docs/assets/icons.js +18 -0
  112. package/docs/assets/icons.svg +1 -0
  113. package/docs/assets/main.js +60 -0
  114. package/docs/assets/navigation.js +1 -0
  115. package/docs/assets/search.js +1 -0
  116. package/docs/assets/style.css +1448 -0
  117. package/docs/classes/src_crate_builder_CrateManager_crate_manager.CrateManager.html +240 -0
  118. package/docs/classes/src_crate_builder_CrateManager_profile_manager.ProfileManager.html +42 -0
  119. package/docs/classes/src_crate_builder_editor_state.EditorState.html +28 -0
  120. package/docs/classes/src_crate_builder_types.CrateManagerType.html +57 -0
  121. package/docs/classes/src_crate_builder_types.ProfileManagerType.html +13 -0
  122. package/docs/functions/src_crate_builder_CrateManager_lib.isURL.html +2 -0
  123. package/docs/functions/src_crate_builder_CrateManager_lib.mintNewCrate.html +3 -0
  124. package/docs/functions/src_crate_builder_CrateManager_lib.normalise.html +5 -0
  125. package/docs/functions/src_crate_builder_CrateManager_lib.normaliseEntityType.html +1 -0
  126. package/docs/index.html +58 -0
  127. package/docs/interfaces/src_crate_builder_types.NormalisedCrate.html +3 -0
  128. package/docs/interfaces/src_crate_builder_types.NormalisedEntityDefinition.html +4 -0
  129. package/docs/interfaces/src_crate_builder_types.NormalisedProfile.html +9 -0
  130. package/docs/interfaces/src_crate_builder_types.ProfileLayout.html +2 -0
  131. package/docs/interfaces/src_crate_builder_types.ProfileLayoutGroup.html +9 -0
  132. package/docs/interfaces/src_crate_builder_types.UnverifiedCrate.html +3 -0
  133. package/docs/interfaces/src_crate_builder_types.UnverifiedEntityDefinition.html +4 -0
  134. package/docs/modules/src_crate_builder_CrateManager_crate_manager.html +2 -0
  135. package/docs/modules/src_crate_builder_CrateManager_lib.html +6 -0
  136. package/docs/modules/src_crate_builder_CrateManager_profile_manager.html +2 -0
  137. package/docs/modules/src_crate_builder_editor_state.html +2 -0
  138. package/docs/modules/src_crate_builder_types.html +16 -0
  139. package/docs/types/src_crate_builder_types.EntityReference.html +1 -0
  140. package/docs/types/src_crate_builder_types.NormalisedContext.html +1 -0
  141. package/docs/types/src_crate_builder_types.PrimitiveType.html +1 -0
  142. package/docs/types/src_crate_builder_types.ProfileAssociation.html +1 -0
  143. package/docs/types/src_crate_builder_types.ProfileInput.html +1 -0
  144. package/docs/types/src_crate_builder_types.UnverifiedContext.html +1 -0
  145. package/docs/variables/src_crate_builder_CrateManager_lib.urlProtocols.html +1 -0
  146. package/index.html +13 -0
  147. package/load-data-packs.cjs +38 -0
  148. package/package.json +135 -0
  149. package/postcss.config.cjs +6 -0
  150. package/public/favicon.ico +0 -0
  151. package/public/index.html +43 -0
  152. package/public/logo192.png +0 -0
  153. package/public/logo512.png +0 -0
  154. package/public/manifest.json +25 -0
  155. package/public/marker-icon.png +0 -0
  156. package/public/marker-shadow.png +0 -0
  157. package/public/robots.txt +3 -0
  158. package/react-app-env.d.ts +1 -0
  159. package/rollup.config.js +26 -0
  160. package/src/app/App.tsx +13 -0
  161. package/src/app/EmbeddedComponent.tsx +432 -0
  162. package/src/app/index.html +20 -0
  163. package/src/app/index.tsx +19 -0
  164. package/src/app/lookup.ts +141 -0
  165. package/src/app/override-styles.css +96 -0
  166. package/src/crate-builder/CrateManager/contexts/1.1-context.jsonld +2660 -0
  167. package/src/crate-builder/CrateManager/contexts/1.2-DRAFT-context.jsonld +2918 -0
  168. package/src/crate-builder/CrateManager/contexts.ts +42 -0
  169. package/src/crate-builder/CrateManager/crate-manager-benchmarking.spec.ts +31 -0
  170. package/src/crate-builder/CrateManager/crate-manager-loading-exporting.spec.ts +431 -0
  171. package/src/crate-builder/CrateManager/crate-manager-operations.spec.ts +298 -0
  172. package/src/crate-builder/CrateManager/crate-manager.spec.ts +2336 -0
  173. package/src/crate-builder/CrateManager/crate-manager.ts +2111 -0
  174. package/src/crate-builder/CrateManager/lib.spec.ts +133 -0
  175. package/src/crate-builder/CrateManager/lib.ts +170 -0
  176. package/src/crate-builder/CrateManager/profile-manager.spec.ts +593 -0
  177. package/src/crate-builder/CrateManager/profile-manager.ts +367 -0
  178. package/src/crate-builder/CrateManager/schema-type-definitions.json +35122 -0
  179. package/src/crate-builder/CrateManager/validate-identifier.spec.ts +82 -0
  180. package/src/crate-builder/CrateManager/validate-identifier.ts +65 -0
  181. package/src/crate-builder/RenderEntity/Add.tsx +249 -0
  182. package/src/crate-builder/RenderEntity/AddControl.stories.tsx +126 -0
  183. package/src/crate-builder/RenderEntity/AddControl.tsx +84 -0
  184. package/src/crate-builder/RenderEntity/AutoComplete.tsx +215 -0
  185. package/src/crate-builder/RenderEntity/BulkAdd.tsx +136 -0
  186. package/src/crate-builder/RenderEntity/DeleteProperty.tsx +33 -0
  187. package/src/crate-builder/RenderEntity/DialogAddProperty.tsx +83 -0
  188. package/src/crate-builder/RenderEntity/DialogBrowseEntities.tsx +136 -0
  189. package/src/crate-builder/RenderEntity/DialogEditContext.tsx +107 -0
  190. package/src/crate-builder/RenderEntity/DialogPreviewCrate.tsx +54 -0
  191. package/src/crate-builder/RenderEntity/DialogSaveCrateAsTemplate.tsx +65 -0
  192. package/src/crate-builder/RenderEntity/DialogSaveEntityTemplate.tsx +87 -0
  193. package/src/crate-builder/RenderEntity/DisplayPropertyName.stories.tsx +30 -0
  194. package/src/crate-builder/RenderEntity/DisplayPropertyName.tsx +21 -0
  195. package/src/crate-builder/RenderEntity/EntityId.tsx +75 -0
  196. package/src/crate-builder/RenderEntity/EntityName.tsx +49 -0
  197. package/src/crate-builder/RenderEntity/EntityProperty.tsx +188 -0
  198. package/src/crate-builder/RenderEntity/EntityPropertyInstance.tsx +255 -0
  199. package/src/crate-builder/RenderEntity/EntityType.tsx +95 -0
  200. package/src/crate-builder/RenderEntity/ItemLink.tsx +37 -0
  201. package/src/crate-builder/RenderEntity/PaginateLinkedEntities.stories.tsx +43 -0
  202. package/src/crate-builder/RenderEntity/PaginateLinkedEntities.tsx +141 -0
  203. package/src/crate-builder/RenderEntity/PropertyHelp.tsx +39 -0
  204. package/src/crate-builder/RenderEntity/RenderControls.tsx +278 -0
  205. package/src/crate-builder/RenderEntity/RenderLinkedItem.tsx +139 -0
  206. package/src/crate-builder/RenderEntity/RenderPropertyHelp.tsx +41 -0
  207. package/src/crate-builder/RenderEntity/RenderReverseConnections.tsx +150 -0
  208. package/src/crate-builder/RenderEntity/RenderTypes.tsx +102 -0
  209. package/src/crate-builder/RenderEntity/Shell2.tsx +576 -0
  210. package/src/crate-builder/RenderEntity/UnlinkEntity.tsx +30 -0
  211. package/src/crate-builder/RenderEntity/auto-complete.lib.ts +184 -0
  212. package/src/crate-builder/RenderEntity/keys.ts +4 -0
  213. package/src/crate-builder/RenderEntity/layout.spec.js +593 -0
  214. package/src/crate-builder/RenderEntity/layout.ts +220 -0
  215. package/src/crate-builder/Shell.tsx +323 -0
  216. package/src/crate-builder/component.css +65 -0
  217. package/src/crate-builder/editor-state.ts +114 -0
  218. package/src/crate-builder/helpers.ts +16 -0
  219. package/src/crate-builder/i18n.ts +22 -0
  220. package/src/crate-builder/lib/validate-iri.js +69 -0
  221. package/src/crate-builder/lib/validate-iri.ts +57 -0
  222. package/src/crate-builder/locales/en.js +149 -0
  223. package/src/crate-builder/locales/hu.js +147 -0
  224. package/src/crate-builder/primitives/Boolean.stories.tsx +33 -0
  225. package/src/crate-builder/primitives/Boolean.tsx +67 -0
  226. package/src/crate-builder/primitives/Date.stories.tsx +32 -0
  227. package/src/crate-builder/primitives/Date.tsx +58 -0
  228. package/src/crate-builder/primitives/DateTime.stories.tsx +32 -0
  229. package/src/crate-builder/primitives/DateTime.tsx +64 -0
  230. package/src/crate-builder/primitives/Geo.stories.tsx +57 -0
  231. package/src/crate-builder/primitives/Geo.tsx +225 -0
  232. package/src/crate-builder/primitives/Map.SelectArea.js +359 -0
  233. package/src/crate-builder/primitives/Map.stories.tsx +61 -0
  234. package/src/crate-builder/primitives/Map.tsx +124 -0
  235. package/src/crate-builder/primitives/Number.stories.tsx +74 -0
  236. package/src/crate-builder/primitives/Number.tsx +166 -0
  237. package/src/crate-builder/primitives/Select.stories.tsx +66 -0
  238. package/src/crate-builder/primitives/Select.tsx +122 -0
  239. package/src/crate-builder/primitives/SelectObject.stories.tsx +29 -0
  240. package/src/crate-builder/primitives/SelectObject.tsx +105 -0
  241. package/src/crate-builder/primitives/SelectUrl.stories.tsx +42 -0
  242. package/src/crate-builder/primitives/SelectUrl.tsx +110 -0
  243. package/src/crate-builder/primitives/Text.stories.tsx +106 -0
  244. package/src/crate-builder/primitives/Text.tsx +197 -0
  245. package/src/crate-builder/primitives/Time.stories.tsx +38 -0
  246. package/src/crate-builder/primitives/Time.tsx +71 -0
  247. package/src/crate-builder/primitives/Url.stories.tsx +43 -0
  248. package/src/crate-builder/primitives/Url.tsx +75 -0
  249. package/src/crate-builder/primitives/Value.stories.tsx +37 -0
  250. package/src/crate-builder/primitives/Value.tsx +30 -0
  251. package/src/crate-builder/primitives/date-libs.ts +12 -0
  252. package/src/crate-builder/profile-schema.json +145 -0
  253. package/src/crate-builder/property-definitions.ts +78 -0
  254. package/src/crate-builder/store.ts +14 -0
  255. package/src/crate-builder/tailwind.css +7 -0
  256. package/src/crate-builder/types.d.ts +318 -0
  257. package/src/examples/collection/collections-entity-example.json +131 -0
  258. package/src/examples/collection/crate-builder-entity-example.json +33 -0
  259. package/src/examples/item/complex-collection/ro-crate-metadata.json +174 -0
  260. package/src/examples/item/complex-item/ro-crate-metadata.json +769 -0
  261. package/src/examples/item/crate-with-language.json +38 -0
  262. package/src/examples/item/empty/ro-crate-metadata.json +20 -0
  263. package/src/examples/item/item-with-relationship-and-action/ro-crate-metadata.json +66 -0
  264. package/src/examples/item/large-crate/ro-crate-metadata.json +5762 -0
  265. package/src/examples/item/multiple-types/ro-crate-metadata.json +20 -0
  266. package/src/examples/item/ridiculously-big-collection/ro-crate-metadata.json +162977 -0
  267. package/src/examples/profile/aroma.complex.profile.json +11098 -0
  268. package/src/examples/profile/aroma.profile.json +9158 -0
  269. package/src/examples/profile/nyingarn-item-profile.json +426 -0
  270. package/src/examples/profile/profile-to-test-inverse-associations.json +73 -0
  271. package/src/examples/profile/profile-to-test-multiple-types.json +31 -0
  272. package/src/examples/profile/profile-with-all-primitives-and-groups.json +207 -0
  273. package/src/examples/profile/profile-with-all-primitives.json +244 -0
  274. package/src/examples/profile/profile-with-constraints.json +446 -0
  275. package/src/examples/profile/profile-with-resolve.json +57 -0
  276. package/src/examples/profile/vocabulary-creation-profile.json +231 -0
  277. package/src/images.d.ts +5 -0
  278. package/src/index.ts +10 -0
  279. package/src/types.ts +104 -0
  280. package/tailwind.config.js +20 -0
  281. package/tsconfig.app.json +31 -0
  282. package/tsconfig.json +26 -0
  283. package/typedoc.json +11 -0
  284. package/update-deps.sh +4 -0
  285. package/vite-env.d.ts +1 -0
  286. package/vite.config.ts +41 -0
@@ -0,0 +1,220 @@
1
+ import difference from "lodash/difference";
2
+ import orderBy from "lodash/orderBy";
3
+ import uniqBy from "lodash/uniqBy";
4
+ const coreProperties = ["@id", "@type", "@reverse", "name"];
5
+ import type {
6
+ ProfileManagerType,
7
+ ProfileInput,
8
+ ProfileLayout,
9
+ ProfileLayoutGroup,
10
+ NormalisedEntityDefinition,
11
+ } from "../types";
12
+
13
+ export function applyLayout({
14
+ configuration,
15
+ entity,
16
+ extraProperties,
17
+ profileManager: pm,
18
+ }: {
19
+ configuration: { [key: string]: any };
20
+ entity: NormalisedEntityDefinition;
21
+ extraProperties: string[];
22
+ profileManager: ProfileManagerType;
23
+ }): {
24
+ renderTabs: boolean;
25
+ missingRequiredData: boolean;
26
+ entity: NormalisedEntityDefinition;
27
+ tabs: ProfileLayoutGroup[];
28
+ } {
29
+ let missingRequiredData = false;
30
+ let inputs = [];
31
+
32
+ ({ missingRequiredData, entity, inputs } = getProfileInputs({
33
+ configuration,
34
+ entity,
35
+ pm,
36
+ }));
37
+
38
+ if (extraProperties.length) {
39
+ extraProperties.forEach((property) => {
40
+ if (!entity[property]) entity[property] = [];
41
+ });
42
+ }
43
+ let layouts = pm.getLayout({ entity });
44
+ let renderTabs = false;
45
+ let tabs = [] as ProfileLayoutGroup[];
46
+ if (layouts) {
47
+ renderTabs = true;
48
+ tabs = applyLayoutToEntity({ layouts, inputs, entity });
49
+ tabs = applyTabDataIndicators({ configuration, tabs, entity });
50
+ }
51
+
52
+ return { renderTabs, missingRequiredData, entity, tabs };
53
+ }
54
+
55
+ function getProfileInputs({
56
+ configuration,
57
+ entity,
58
+ pm,
59
+ }: {
60
+ configuration: { [key: string]: any };
61
+ entity: NormalisedEntityDefinition;
62
+ pm: ProfileManagerType;
63
+ }): {
64
+ missingRequiredData: boolean;
65
+ entity: NormalisedEntityDefinition;
66
+ inputs: ProfileInput[];
67
+ } {
68
+ const inputs = pm.getInputsFromProfile({ entity });
69
+
70
+ let missingRequiredData = false;
71
+ for (let input of inputs) {
72
+ if (input.name === "name") continue;
73
+ if (!input.id) {
74
+ console.error(`Excluding invalid input - missing id: ${input}`);
75
+ continue;
76
+ }
77
+ if (!entity[input.name] && !configuration.readonly) {
78
+ entity[input.name] = [];
79
+ }
80
+
81
+ if (input.required && (!entity[input.name] || !entity[input.name].length)) {
82
+ missingRequiredData = true;
83
+ }
84
+ }
85
+ return { missingRequiredData, entity, inputs };
86
+ }
87
+
88
+ function applyLayoutToEntity({
89
+ layouts,
90
+ inputs,
91
+ entity,
92
+ }: {
93
+ layouts: ProfileLayout;
94
+ inputs: ProfileInput[];
95
+ entity: NormalisedEntityDefinition;
96
+ }): ProfileLayoutGroup[] {
97
+ // normalise the layouts
98
+ let sort = false;
99
+ for (let name of Object.keys(layouts)) {
100
+ layouts[name].name = name;
101
+ layouts[name].inputs = [];
102
+ if (!layouts[name].properties) layouts[name].properties = [];
103
+ if (layouts[name].order) sort = true;
104
+ }
105
+
106
+ // sort the profile inputs into their groups
107
+ for (let input of inputs) {
108
+ if (coreProperties.includes(input.name)) continue;
109
+ if (!entity[input.name]) continue;
110
+ if (input.hide) continue;
111
+ if (input.group && layouts[input.group]) {
112
+ layouts[input.group].inputs.push(input);
113
+ }
114
+ }
115
+
116
+ // now sort any ungrouped inputs using the properties
117
+ // array in each layout
118
+ let ungroupedInputs = [];
119
+ for (let input of inputs) {
120
+ if (coreProperties.includes(input.name)) continue;
121
+ if (!entity[input.name]) continue;
122
+ if (input?.hide) continue;
123
+ if (input.group && layouts[input.group]) continue;
124
+ let matched = false;
125
+ for (let name of Object.keys(layouts)) {
126
+ if (
127
+ layouts[name] &&
128
+ layouts[name].properties &&
129
+ (layouts[name].properties.includes(input.id) ||
130
+ layouts[name].properties.includes(input.name))
131
+ ) {
132
+ layouts[name].inputs.push(input);
133
+ matched = true;
134
+ }
135
+ }
136
+ if (!matched) ungroupedInputs.push(input);
137
+ }
138
+
139
+ // and any that still aren't part of a group go into overflow
140
+ for (let input of ungroupedInputs) {
141
+ layouts.overflow.inputs.push(input);
142
+ }
143
+
144
+ // get a list of the properties defined on the entity
145
+ // but which have no input definition and pop them
146
+ // into the overflow group with a default Text configuration
147
+ let profileInputs = inputs.map((i) => i.name);
148
+ let entityProperties = Object.keys(entity);
149
+ let missingInputs = difference(entityProperties, profileInputs);
150
+ for (let input of missingInputs) {
151
+ if (coreProperties.includes(input)) continue;
152
+ layouts.overflow.inputs.push({
153
+ id: "",
154
+ name: input,
155
+ multiple: true,
156
+ type: ["Text"],
157
+ });
158
+ }
159
+
160
+ let tabs: any[] = Object.keys(layouts)
161
+ .map((k) => layouts[k])
162
+ .filter((t) => t.name !== "appliesTo");
163
+ if (sort) tabs = orderBy(tabs, "order");
164
+
165
+ // if a properties array is defined
166
+ // reorder the inputs by the order in that array
167
+ for (let tab of tabs) {
168
+ if (tab.properties.length) {
169
+ let orderedInputs = [];
170
+ let otherInputs = [];
171
+ for (let property of tab.properties) {
172
+ for (let input of tab.inputs) {
173
+ if ([input.name, input.id].includes(property)) {
174
+ orderedInputs.push(input);
175
+ } else {
176
+ otherInputs.push(input);
177
+ }
178
+ }
179
+ }
180
+ orderedInputs = uniqBy(orderedInputs, "name");
181
+ otherInputs = uniqBy(otherInputs, "name");
182
+ otherInputs = difference(otherInputs, orderedInputs);
183
+ tab.inputs = [...orderedInputs, ...otherInputs];
184
+ }
185
+ }
186
+ return tabs;
187
+ }
188
+
189
+ function applyTabDataIndicators({
190
+ configuration,
191
+ tabs,
192
+ entity,
193
+ }: {
194
+ configuration: { [key: string]: any };
195
+ tabs: ProfileLayoutGroup[];
196
+ entity: NormalisedEntityDefinition;
197
+ }): ProfileLayoutGroup[] {
198
+ // special case: component in readonly mode
199
+ if (configuration.readonly) {
200
+ tabs = tabs.map((tab) => {
201
+ tab.hasData = false;
202
+ tab.missingRequiredData = false;
203
+ return tab;
204
+ });
205
+ } else {
206
+ for (let tab of tabs) {
207
+ tab.missingRequiredData = false;
208
+ tab.hasData = false;
209
+ for (let input of tab.inputs) {
210
+ if (input.required && !entity[input.name].length) {
211
+ tab.missingRequiredData = true;
212
+ }
213
+ if (entity[input.name].length) {
214
+ tab.hasData = true;
215
+ }
216
+ }
217
+ }
218
+ }
219
+ return tabs;
220
+ }
@@ -0,0 +1,323 @@
1
+ import React, {
2
+ useState,
3
+ useEffect,
4
+ useRef,
5
+ useCallback,
6
+ useMemo,
7
+ useContext,
8
+ JSX,
9
+ } from "react";
10
+ //import RenderEntity from "./RenderEntity/Shell";
11
+ import RenderEntity from "./RenderEntity/Shell2";
12
+ import {
13
+ CrateManager,
14
+ } from "./CrateManager/crate-manager";
15
+ import {
16
+ ProfileManager,
17
+ } from "./CrateManager/profile-manager";
18
+ import { useStateStore } from "./store";
19
+ import i18next from "./i18n";
20
+ import isEmpty from "lodash/isEmpty";
21
+ import isFunction from "lodash/isFunction";
22
+ import { CrateManagerType, ProfileManagerType, UnverifiedCrate } from "./types"; // Add appropriate types
23
+ import { propertyDefinitions } from "./property-definitions";
24
+ import { Lookup } from "./RenderEntity/auto-complete.lib";
25
+
26
+ export interface DescriboCrateBuilderProps {
27
+ crate?: CrateManagerType["crate"];
28
+ profile?: ProfileManagerType["profile"];
29
+ entityId?: string;
30
+ lookup?: any;
31
+ enableContextEditor?: boolean;
32
+ enableCratePreview?: boolean;
33
+ enableBrowseEntities?: boolean;
34
+ enableTemplateSave?: boolean;
35
+ enableReverseLinkBrowser?: boolean;
36
+ enableUrlMarkup?: boolean;
37
+ enableEntityTimestamps?: boolean;
38
+ purgeUnlinkedEntities?: boolean;
39
+ readonly?: boolean;
40
+ webComponent?: boolean;
41
+ tabLocation?: string;
42
+ showControls?: boolean;
43
+ language?: string;
44
+ resetTabOnEntityChange?: boolean;
45
+ resetTabOnProfileChange?: boolean;
46
+ onReady?: () => void;
47
+ onError?: (errors: any) => void;
48
+ onWarning?: (warnings: any) => void;
49
+ onNavigation?: (info: { "@id": string; "@type": string[]; name: string }) => void;
50
+ onSaveCrate?: (saveData: {crate: any}) => void,
51
+ onSaveEntityTemplate?: (entity: any) => void;
52
+ }
53
+
54
+
55
+ // Use propertyDefinitions to define props
56
+ export const ProfileManagerContext = React.createContext<any>(null);
57
+ export const CrateManagerContext = React.createContext<any>(null);
58
+ export const LookupsContext = React.createContext<any>(null);
59
+
60
+ export function DescriboCrateBuilder(props: DescriboCrateBuilderProps): JSX.Element {
61
+ const {
62
+ crate,
63
+ profile,
64
+ entityId,
65
+ lookup,
66
+ enableContextEditor = propertyDefinitions.enableContextEditor.default,
67
+ enableCratePreview = propertyDefinitions.enableCratePreview.default,
68
+ enableBrowseEntities = propertyDefinitions.enableBrowseEntities.default,
69
+ enableTemplateSave = propertyDefinitions.enableTemplateSave.default,
70
+ enableReverseLinkBrowser = propertyDefinitions.enableReverseLinkBrowser.default,
71
+ enableUrlMarkup = propertyDefinitions.enableUrlMarkup.default,
72
+ enableEntityTimestamps = propertyDefinitions.enableEntityTimestamps.default,
73
+ purgeUnlinkedEntities = propertyDefinitions.purgeUnlinkedEntities.default,
74
+ readonly = propertyDefinitions.readonly.default,
75
+ webComponent = propertyDefinitions.webComponent.default,
76
+ tabLocation = propertyDefinitions.tabLocation.default,
77
+ showControls = propertyDefinitions.showControls.default,
78
+ language = propertyDefinitions.language.default,
79
+ resetTabOnEntityChange = false,
80
+ resetTabOnProfileChange = false,
81
+ onReady,
82
+ onError,
83
+ onWarning,
84
+ onNavigation,
85
+ onSaveCrate,
86
+ onSaveEntityTemplate,
87
+ } = props;
88
+
89
+ console.log("DescriboCrateBuilder props:", { crate, profile, entityId });
90
+ const [ready, setReady] = useState(false);
91
+ const [error, setError] = useState(false);
92
+ const [contextEntity, setContextEntity] = useState<any | null>(null);
93
+ const [cm, setCm] = useState<CrateManager | null>(null);
94
+ const [pm, setPm] = useState<ProfileManager | null>(null);
95
+ const [lookups, setLookups] = useState<typeof propertyDefinitions.lookup | null>(null);
96
+ const state = useStateStore();
97
+ const renderEntityRef = useRef<any>(null);
98
+ const keys = useRef({ cm: 0, pm: 0, lookups: 0 });
99
+
100
+ const configuration = useMemo(() => {
101
+ const config = {
102
+ enableContextEditor,
103
+ enableCratePreview,
104
+ enableBrowseEntities,
105
+ enableTemplateSave,
106
+ enableReverseLinkBrowser,
107
+ enableUrlMarkup,
108
+ enableEntityTimestamps,
109
+ purgeUnlinkedEntities,
110
+ readonly,
111
+ webComponent,
112
+ tabLocation,
113
+ showControls,
114
+ language,
115
+ enableTemplateLookups: false,
116
+ enableDataPackLookups: false,
117
+ };
118
+
119
+
120
+
121
+ if (lookup) {
122
+ if (isFunction(lookup.entityTemplates)) {
123
+ config.enableTemplateLookups = true;
124
+ }
125
+ if (isFunction(lookup.dataPacks)) {
126
+ config.enableDataPackLookups = true;
127
+ }
128
+ }
129
+
130
+ return config;
131
+ }, [
132
+ enableContextEditor,
133
+ enableCratePreview,
134
+ enableBrowseEntities,
135
+ enableTemplateSave,
136
+ enableReverseLinkBrowser,
137
+ enableUrlMarkup,
138
+ enableEntityTimestamps,
139
+ purgeUnlinkedEntities,
140
+ readonly,
141
+ webComponent,
142
+ tabLocation,
143
+ showControls,
144
+ language,
145
+ lookup,
146
+ ]);
147
+
148
+ const init = useCallback(async () => {
149
+
150
+ console.log("Initializing with crate:", crate);
151
+ console.log("Profile loaded:", profile);
152
+
153
+ i18next.changeLanguage(language);
154
+
155
+ if (!crate || isEmpty(crate)) {
156
+ setReady(false);
157
+ setContextEntity(null);
158
+ setError(false);
159
+ return;
160
+ }
161
+
162
+ const newCm = new CrateManager({
163
+ crate: crate as UnverifiedCrate,
164
+ entityTimestamps: enableEntityTimestamps,
165
+ });
166
+
167
+ console.log("Entity:", newCm.getEntity({ id: entityId || "./" }));
168
+
169
+ // Create a deep clone of the profile to avoid mutation issues
170
+ const profileClone = profile ? JSON.parse(JSON.stringify(profile)) : {};
171
+
172
+ const newPm = new ProfileManager({
173
+ profile: profileClone,
174
+ });
175
+
176
+ // Add a key property to the ProfileManager for tracking changes
177
+ newPm.$key = ++keys.current.pm;
178
+
179
+ newCm.setProfileManager(newPm as ProfileManagerType);
180
+
181
+ // Only set context if it exists
182
+ if (newPm.context) newCm.setContext(newPm.context);
183
+
184
+ if (lookup) {
185
+ setLookups({ ...lookup, $key: ++keys.current.lookups });
186
+ }
187
+
188
+ const errors = newCm.getErrors();
189
+ if (errors.hasError) {
190
+ onError?.({ errors }); // Pass errors wrapped in an object to match Vue implementation
191
+ setReady(false);
192
+ //setError(true);
193
+ console.error("❌ CrateManager reported errors:", errors);
194
+ return;
195
+ }
196
+
197
+ const warnings = newCm.getWarnings();
198
+ if (warnings.hasWarning) {
199
+ onWarning?.({ warnings }); // Pass warnings wrapped in an object
200
+ }
201
+
202
+ setCm(newCm);
203
+ setPm(newPm);
204
+ setError(false);
205
+
206
+ // Force state update to ensure UI refreshes
207
+ state.configuration = configuration;
208
+
209
+ try {
210
+ // Add error handling for entity setting
211
+ if (entityId) {
212
+ handleSetCurrentEntity({ id: entityId });
213
+ } else {
214
+ // Make sure root entity exists and can be set
215
+ const rootEntity = newCm.getEntity({ id: "./" });
216
+ if (!isEmpty(rootEntity)) {
217
+ setContextEntity(rootEntity);
218
+ state.editorState.push({ id: rootEntity["@id"] });
219
+ } else {
220
+ console.warn("Root entity './' not found");
221
+ setError(true);
222
+ return;
223
+ }
224
+ }
225
+
226
+ setReady(true);
227
+ onReady?.();
228
+ } catch (error) {
229
+ console.error("Error setting current entity:", error);
230
+ setError(true);
231
+ onError?.({ error: "Failed to initialize entity" });
232
+ }
233
+ }, [crate, profile, entityId, lookup, configuration]);
234
+
235
+ const handleSetCurrentEntity = useCallback((params: { id: string; updateState?: boolean }) => {
236
+ const { id, updateState = true } = params;
237
+ if (!id || !cm) return;
238
+
239
+ try {
240
+ const entity = cm.getEntity({ id });
241
+ if (!isEmpty(entity)) {
242
+ setContextEntity(entity);
243
+ onNavigation?.({
244
+ "@id": entity["@id"],
245
+ "@type": entity["@type"],
246
+ name: entity.name,
247
+ });
248
+
249
+ if (updateState) {
250
+ state.editorState.push({ id: entity["@id"] });
251
+ }
252
+ } else {
253
+ console.warn(`Entity with id ${id} not found`);
254
+ // Try to fall back to root entity if not already trying to load it
255
+ if (id !== "./") {
256
+ handleSetCurrentEntity({ id: "./" });
257
+ }
258
+ }
259
+ } catch (error) {
260
+ console.error("Error in handleSetCurrentEntity:", error);
261
+ onError?.({ error: "Failed to set entity" });
262
+ }
263
+ }, [cm, onNavigation]);
264
+
265
+ const saveEntityAsTemplate = () => {
266
+ if (!cm || !contextEntity) return;
267
+ const template = cm.exportEntityTemplate({
268
+ id: contextEntity["@id"],
269
+ resolveDepth: 1, // Default resolveDepth or adjust as needed
270
+ });
271
+ onSaveEntityTemplate?.(template);
272
+ };
273
+
274
+ const handleWarning = () => {
275
+ onWarning?.({});
276
+ };
277
+
278
+ const handleError = () => {
279
+ onError?.({});
280
+ };
281
+
282
+ useEffect(() => {
283
+ state.editorState.reset();
284
+ init();
285
+ }, [crate, profile, init]);
286
+
287
+ useEffect(() => {
288
+ if (entityId && entityId !== contextEntity?.["@id"]) {
289
+ handleSetCurrentEntity({ id: entityId }); // Pass an object with the id property
290
+ }
291
+ }, [entityId, contextEntity, handleSetCurrentEntity]);
292
+
293
+ const saveCrate = () => {
294
+ if (!cm) return;
295
+ if (purgeUnlinkedEntities) {
296
+ cm.purgeUnlinkedEntities();
297
+ }
298
+ const exported = cm.exportCrate();
299
+ onSaveCrate?.({ crate: exported });
300
+ };
301
+
302
+ return (
303
+ <ProfileManagerContext.Provider value={pm}>
304
+ <CrateManagerContext.Provider value={cm}>
305
+ <LookupsContext.Provider value={lookups}>
306
+ {ready && !error && contextEntity && (
307
+ <RenderEntity
308
+ ref={renderEntityRef}
309
+ entity={contextEntity}
310
+ onLoadEntity={handleSetCurrentEntity}
311
+ onSaveCrate={saveCrate}
312
+ onSaveEntityTemplate={saveEntityAsTemplate}
313
+ onWarning={handleWarning}
314
+ onError={handleError}
315
+ />
316
+ )}
317
+ </LookupsContext.Provider>
318
+ </CrateManagerContext.Provider>
319
+ </ProfileManagerContext.Provider>
320
+ );
321
+ };
322
+
323
+ export default DescriboCrateBuilder;
@@ -0,0 +1,65 @@
1
+ .el-tabs__item {
2
+ height: unset;
3
+ }
4
+
5
+ .el-select-dropdown__wrap {
6
+ max-height: 500px !important;
7
+ }
8
+
9
+ .el-select-group__title {
10
+ font-size: 14px !important;
11
+ }
12
+
13
+ /* ::-webkit-scrollbar {
14
+ display: none;
15
+ } */
16
+
17
+ /* Customize website's scrollbar like Mac OS
18
+ Not supports in Firefox and IE */
19
+ .scrollbar {
20
+ overflow: overlay
21
+ }
22
+
23
+ /* total width */
24
+ .scrollbar::-webkit-scrollbar {
25
+ background-color: rgba(0, 0, 0, 0);
26
+ width: 16px;
27
+ height: 16px;
28
+ z-index: 999999;
29
+ }
30
+
31
+ /* background of the scrollbar except button or resizer */
32
+ .scrollbar::-webkit-scrollbar-track {
33
+ background-color: rgba(0, 0, 0, 0);
34
+ }
35
+
36
+ /* scrollbar itself */
37
+ .scrollbar::-webkit-scrollbar-thumb {
38
+ background-color: rgba(0, 0, 0, 0);
39
+ border-radius: 16px;
40
+ border: 0px solid #fff;
41
+ }
42
+
43
+ /* set button(top and bottom of the scrollbar) */
44
+ .scrollbar::-webkit-scrollbar-button {
45
+ display: none;
46
+ }
47
+
48
+
49
+ /* scrollbar when element is hovered */
50
+ .scrollbar:hover::-webkit-scrollbar-thumb {
51
+ background-color: #a0a0a5;
52
+ border: 4px solid #fff;
53
+ }
54
+
55
+ /* scrollbar when scrollbar is hovered */
56
+ .scrollbar::-webkit-scrollbar-thumb:hover {
57
+ background-color: #a0a0a5;
58
+ border: 4px solid #f4f4f4
59
+ }
60
+
61
+ .cm-editor {
62
+ font-size: 12px;
63
+ height: 350px;
64
+ overflow: auto;
65
+ }