keycloak-scim/js/apps/account-ui/src/content/ContentComponent.tsx
Erik Jan de Wit f20de396aa
better path resolve (#26239)
for https://github.com/keycloak/keycloak-quickstarts/pull/523

Signed-off-by: Erik Jan de Wit <erikjan.dewit@gmail.com>
2024-01-19 10:46:08 +01:00

58 lines
1.5 KiB
TypeScript

import { Spinner } from "@patternfly/react-core";
import { Suspense, lazy, useMemo, useState } from "react";
import { useParams } from "react-router-dom";
import { environment } from "../environment";
import { ContentComponentParams } from "../routes";
import { joinPath } from "../utils/joinPath";
import { usePromise } from "../utils/usePromise";
import fetchContentJson from "./fetchContent";
import { MenuItem } from "../root/PageNav";
function findComponent(
content: MenuItem[],
componentId: string,
): string | undefined {
for (const item of content) {
if (
"path" in item &&
item.path.endsWith(componentId) &&
"modulePath" in item
) {
return item.modulePath;
}
if ("children" in item) {
return findComponent(item.children, componentId);
}
}
return undefined;
}
const ContentComponent = () => {
const [content, setContent] = useState<MenuItem[]>();
const { componentId } = useParams<ContentComponentParams>();
usePromise((signal) => fetchContentJson({ signal }), setContent);
const modulePath = useMemo(
() => findComponent(content || [], componentId!),
[content, componentId],
);
return (
<Suspense fallback={<Spinner />}>
{modulePath && <Component modulePath={modulePath} />}
</Suspense>
);
};
type ComponentProps = {
modulePath: string;
};
const Component = ({ modulePath }: ComponentProps) => {
const Element = lazy(
() => import(joinPath(environment.resourceUrl, modulePath)),
);
return <Element />;
};
export default ContentComponent;