2022-04-23 20:35:53 -04:00
|
|
|
import style from '../styles/lists.module.css';
|
|
|
|
import React, { ReactElement } from 'react';
|
|
|
|
|
2022-09-27 20:52:24 -04:00
|
|
|
export interface listItem {
|
|
|
|
[x: string]: any;
|
2022-04-23 20:35:53 -04:00
|
|
|
children?: listItem[] | string[];
|
|
|
|
url?: string;
|
2022-09-27 20:52:24 -04:00
|
|
|
type?: string;
|
2022-04-23 20:35:53 -04:00
|
|
|
title: string;
|
|
|
|
description?: string;
|
|
|
|
};
|
|
|
|
|
|
|
|
export function toListItem(record: Record<string, any>): listItem | null {
|
|
|
|
if (!record.title)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
let children: listItem[] | string[] = [];
|
|
|
|
if (Array.isArray(record.children) && record.children.length) {
|
|
|
|
|
|
|
|
let lchildren: listItem[] = [];
|
|
|
|
let schildren: string[] = [];
|
|
|
|
for (const child of record.children) {
|
|
|
|
if (typeof child === 'string') {
|
|
|
|
schildren.push(child);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const lChild = toListItem(child);
|
|
|
|
if (lChild)
|
|
|
|
lchildren.push(lChild);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!lchildren.length) {
|
|
|
|
children = schildren;
|
|
|
|
}
|
|
|
|
else {
|
2022-04-24 00:27:51 -04:00
|
|
|
children = [...lchildren, ...schildren.map((s: string): listItem => {
|
2022-04-23 20:35:53 -04:00
|
|
|
return { title: s };
|
2022-04-24 00:27:51 -04:00
|
|
|
})];
|
2022-04-23 20:35:53 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-27 20:52:24 -04:00
|
|
|
return Object.assign(record, {
|
2022-04-23 20:35:53 -04:00
|
|
|
title: record.title,
|
|
|
|
url: record.url,
|
2022-04-24 00:27:51 -04:00
|
|
|
children: children.length ? children : undefined,
|
2022-09-27 20:52:24 -04:00
|
|
|
type: record.type?.length ? record.type : undefined,
|
2022-04-23 20:35:53 -04:00
|
|
|
description: record.description,
|
2022-09-27 20:52:24 -04:00
|
|
|
});
|
2022-04-23 20:35:53 -04:00
|
|
|
}
|
|
|
|
|
2022-09-27 20:52:24 -04:00
|
|
|
export function mapChild(
|
|
|
|
obj: listItem | string,
|
|
|
|
level: number,
|
|
|
|
typeMap? : Record<string, (o: listItem) => JSX.Element>
|
|
|
|
) {
|
2022-04-23 20:35:53 -04:00
|
|
|
if (typeof obj === 'string') {
|
|
|
|
if (obj === '')
|
|
|
|
return <></>
|
|
|
|
return <span className={style.listItem}>{obj}</span>
|
|
|
|
}
|
|
|
|
|
|
|
|
if (obj.title === '')
|
|
|
|
return <></>
|
|
|
|
|
2022-04-27 05:36:54 -04:00
|
|
|
const desc = obj.description
|
|
|
|
? <span className={style.listItemDesc}>{obj.description}</span>
|
|
|
|
: <></>;
|
2022-09-27 20:52:24 -04:00
|
|
|
|
2022-04-23 20:35:53 -04:00
|
|
|
if (obj.url)
|
2022-04-27 05:36:54 -04:00
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<span className={style.listItem}><a href={obj.url}>{obj.title}</a></span>
|
|
|
|
{desc}
|
|
|
|
</>);
|
2022-04-23 20:35:53 -04:00
|
|
|
|
2022-09-27 20:52:24 -04:00
|
|
|
if (!obj.children) {
|
|
|
|
let cb;
|
|
|
|
if (obj.type && typeMap) {
|
|
|
|
cb = typeMap[obj.type]
|
|
|
|
}
|
|
|
|
|
|
|
|
return cb
|
|
|
|
? cb(obj)
|
|
|
|
: (<><span className={style.listItem}>{obj.title}</span>{desc}</>);
|
|
|
|
}
|
2022-04-23 20:35:53 -04:00
|
|
|
|
|
|
|
let title: ReactElement;
|
|
|
|
|
|
|
|
if (level >= 0 && level <= 4)
|
|
|
|
title = React.createElement(`h${level + 2}`, {}, obj.title);
|
|
|
|
else
|
|
|
|
title = React.createElement('strong', {}, obj.title);
|
|
|
|
|
|
|
|
return (
|
2022-04-27 21:55:18 -04:00
|
|
|
<section className={level < 4 && `block ${style.block}` || ''}>
|
2022-04-23 20:35:53 -04:00
|
|
|
{title}
|
2022-04-27 05:36:54 -04:00
|
|
|
{obj.description ? <p className={style.desc}>{obj.description}</p> : <></>}
|
2022-04-23 20:35:53 -04:00
|
|
|
<div>
|
2022-09-27 20:52:24 -04:00
|
|
|
{obj.children.map(l => mapChild(l, level + 1, typeMap))}
|
2022-04-23 20:35:53 -04:00
|
|
|
</div>
|
2022-04-27 05:10:49 -04:00
|
|
|
</section>
|
2022-04-23 20:35:53 -04:00
|
|
|
);
|
2023-10-30 00:18:38 -04:00
|
|
|
}
|