-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathmoduleLoader.tsx
69 lines (61 loc) · 1.7 KB
/
moduleLoader.tsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
import React from "react";
type Module = any;
type ModuleArray = { name: string; content: any }[];
type ModuleMap = { [moduleName: string]: Module };
type ModuleImporter = () => Promise<Module>;
const ModuleLoaderContext = React.createContext<ModuleMap | null>(null);
function moduleArrayToMap(modules: ModuleArray) {
return modules.reduce<ModuleMap>(
(moduleMap, module) => ({
...moduleMap,
[module.name]: module.content,
}),
{}
);
}
function ModuleLoader({
imports,
children,
}: {
imports: {
// noinspection JSUnusedLocalSymbols (jetbrains bug)
[name: string]: ModuleImporter;
};
children: React.ReactNode;
}) {
const [modules, setModules] = React.useState<ModuleMap | null>(null);
React.useEffect(() => {
Promise.all(
Object.keys(imports).map(async (importName) => ({
name: "react-flow-renderer",
content: await imports[importName](),
}))
).then(
(modules) => setModules(moduleArrayToMap(modules)),
(err) => {
throw err;
}
);
return () => setModules(null);
}, [imports]);
return (
<ModuleLoaderContext.Provider value={modules}>
{modules != null ? children : "Loading modules"}
</ModuleLoaderContext.Provider>
);
}
function useModuleLoader(importName: string) {
const imports = React.useContext(ModuleLoaderContext);
if (imports == null) {
throw new Error(
"Be sure to use useModuleLoader in a component wrapped in ModuleLoader"
);
}
if (imports[importName] == null) {
throw new Error(
`Cannot find ${importName}, is it declared in the 'imports' prop of the ModuleLoader ?`
);
}
return imports[importName];
}
export { ModuleLoader, useModuleLoader };