164 lines
4.7 KiB
TypeScript
164 lines
4.7 KiB
TypeScript
import Image from "next/image";
|
|
import { getAllWikiPaths, getWikiContent } from "../../../lib/wiki/api";
|
|
import WikiPage from "../../../interfaces/WikiPage";
|
|
import DashLayout from "../../../layouts/DashLayout";
|
|
import Link from "next/link";
|
|
import { m } from "framer-motion";
|
|
import RenderMarkdown from "../../../components/wiki/RenderMarkdown";
|
|
import PageBody from "../../../components/wiki/PageBody";
|
|
import { ReactElement, useEffect, useState } from "react";
|
|
|
|
interface WikiLandingPageProps {
|
|
children: React.ReactNode;
|
|
page: WikiPage;
|
|
}
|
|
|
|
interface TableOfContentsItem {
|
|
id: string;
|
|
type: string;
|
|
text: string;
|
|
}
|
|
|
|
function WikiLandingPage(props: WikiLandingPageProps) {
|
|
const [wikiContent, setWikiContent] = useState<ReactElement>(<></>);
|
|
const [indexContent, setIndexContent] = useState<TableOfContentsItem[]>([]);
|
|
|
|
// needed for proper hydration due to replacing some elements
|
|
useEffect(() => {
|
|
setWikiContent(<PageBody>{props.page.content}</PageBody>);
|
|
}, [props.page.content]);
|
|
|
|
useEffect(() => {
|
|
const toc: TableOfContentsItem[] = [];
|
|
const headings = document.querySelectorAll("h2, h3");
|
|
// store the heading text, id, and type, keep order
|
|
headings.forEach((heading) => {
|
|
const id = heading.getAttribute("id");
|
|
const type = heading.tagName.toLowerCase();
|
|
const text = heading.textContent;
|
|
if (id && type && text) {
|
|
toc.push({ id, type, text });
|
|
}
|
|
});
|
|
setIndexContent(toc);
|
|
console.log(toc);
|
|
}, [wikiContent]);
|
|
|
|
return (
|
|
<div className="flex h-full w-full flex-col items-center justify-center p-3 font-plusJakarta ">
|
|
<div className="inline-grid h-full w-full max-w-screen-2xl grid-cols-10">
|
|
{/* Sidebar */}
|
|
<div className="col-span-2 hidden w-full flex-col items-center justify-center lg:block">
|
|
<div className="w-full rounded-tl-2xl rounded-bl-2xl border-r-2 border-orange-400 border-opacity-60 bg-zinc-800 bg-opacity-70 p-6 text-left text-6xl text-white">
|
|
<div className="text-2xl">Contents</div>
|
|
<div className="mt-4 text-left text-orange-400">
|
|
{indexContent.map((item) => {
|
|
return (
|
|
// increase indent based on heading level
|
|
<div
|
|
style={{
|
|
paddingLeft: `${(parseInt(item.type[1]) - 2) * 1.25}rem`,
|
|
}}
|
|
className="text-xl"
|
|
key={item.id}
|
|
>
|
|
<Link href={`#${item.id}`}>
|
|
<p className="mt-2 overflow-hidden overflow-ellipsis whitespace-nowrap hover:text-white">
|
|
{item.text}
|
|
</p>
|
|
</Link>
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div className="col-span-10 rounded-2xl bg-zinc-800 bg-opacity-70 p-6 text-center text-6xl text-white lg:col-span-8 lg:rounded-tl-none">
|
|
<m.div
|
|
initial={{ opacity: 0 }}
|
|
animate={{ opacity: 1 }}
|
|
transition={{ delay: 0.3 }}
|
|
key={props.page.slug}
|
|
>
|
|
<div className="w-full px-3">{wikiContent}</div>
|
|
</m.div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
type Params = {
|
|
params: {
|
|
slug: string[];
|
|
};
|
|
};
|
|
|
|
export async function getStaticProps({ params }: Params) {
|
|
// only language
|
|
if (params.slug.length < 2) {
|
|
// if langauge is two letters, redirect to its home
|
|
if (params.slug[0].length === 2 && params.slug[0].match(/^[a-z]+$/i)) {
|
|
return {
|
|
redirect: {
|
|
destination: `/wiki/${params.slug[0]}/home`,
|
|
},
|
|
};
|
|
}
|
|
// else, 404 to prevemt building pointless pages (e.g. /wiki/this-is-not-a-language x 580258538 times)
|
|
return {
|
|
notFound: true,
|
|
};
|
|
}
|
|
// slug[0] = language
|
|
// slug[1...n] = page path
|
|
const lang = params.slug[0];
|
|
const path = params.slug.slice(1).join("/");
|
|
const pageData = getWikiContent(lang, path);
|
|
|
|
// if no content, 404
|
|
if (!pageData) {
|
|
return {
|
|
notFound: true,
|
|
};
|
|
}
|
|
|
|
return {
|
|
props: {
|
|
page: {
|
|
slug: params.slug,
|
|
content: pageData.content,
|
|
data: pageData.data,
|
|
},
|
|
},
|
|
};
|
|
}
|
|
|
|
export async function getStaticPaths() {
|
|
const paths = getAllWikiPaths();
|
|
|
|
return {
|
|
paths: paths.map((path) => {
|
|
return {
|
|
params: {
|
|
slug: path.split("/"),
|
|
},
|
|
};
|
|
}),
|
|
fallback: "blocking",
|
|
};
|
|
}
|
|
|
|
WikiLandingPage.getLayout = function getLayout(page: React.ReactNode) {
|
|
const metaTags = {
|
|
title: "Wiki - toffee",
|
|
description: "Wiki for toffee",
|
|
};
|
|
return (
|
|
<DashLayout metaTags={metaTags} navIcon="wiki">
|
|
{page}
|
|
</DashLayout>
|
|
);
|
|
};
|
|
|
|
export default WikiLandingPage;
|