mirror of
https://github.com/ethereum/go-ethereum.git
synced 2026-04-09 19:37:29 +00:00
140 lines
3.9 KiB
TypeScript
140 lines
3.9 KiB
TypeScript
import fs from 'fs';
|
|
import matter from 'gray-matter';
|
|
import yaml from 'js-yaml';
|
|
import { Flex, Stack, Heading, Text } from '@chakra-ui/react';
|
|
import ChakraUIRenderer from 'chakra-ui-markdown-renderer';
|
|
import ReactMarkdown from 'react-markdown';
|
|
import { useRouter } from 'next/router';
|
|
import { useEffect } from 'react';
|
|
import gfm from 'remark-gfm';
|
|
import rehypeRaw from 'rehype-raw';
|
|
import { ParsedUrlQuery } from 'querystring';
|
|
import type { GetStaticPaths, GetStaticProps, NextPage } from 'next';
|
|
|
|
import MDComponents from '../components/UI/docs';
|
|
import { Breadcrumbs, DocsNav, DocumentNav } from '../components/UI/docs';
|
|
import { PageMetadata } from '../components/UI';
|
|
|
|
import { NavLink } from '../types';
|
|
|
|
import { getFileList } from '../utils/getFileList';
|
|
|
|
import { textStyles } from '../theme/foundations';
|
|
import { getParsedDate } from '../utils';
|
|
|
|
const MATTER_OPTIONS = {
|
|
engines: {
|
|
yaml: (s: any) => yaml.load(s, { schema: yaml.JSON_SCHEMA }) as object
|
|
}
|
|
};
|
|
|
|
// This method crawls for all valid docs paths
|
|
export const getStaticPaths: GetStaticPaths = () => {
|
|
const paths: string[] = getFileList('docs'); // This is folder that get crawled for valid docs paths. Change if this path changes.
|
|
|
|
return {
|
|
paths,
|
|
fallback: false
|
|
};
|
|
};
|
|
|
|
// Reads file data for markdown pages
|
|
export const getStaticProps: GetStaticProps = async context => {
|
|
const { slug } = context.params as ParsedUrlQuery;
|
|
const filePath = (slug as string[])!.join('/');
|
|
let file;
|
|
let lastModified;
|
|
|
|
const navLinks = yaml.load(fs.readFileSync('src/data/documentation-links.yaml', 'utf8'));
|
|
|
|
try {
|
|
file = fs.readFileSync(`${filePath}.md`, 'utf-8');
|
|
lastModified = fs.statSync(`${filePath}.md`);
|
|
} catch {
|
|
file = fs.readFileSync(`${filePath}/index.md`, 'utf-8');
|
|
lastModified = fs.statSync(`${filePath}/index.md`);
|
|
}
|
|
|
|
const { data: frontmatter, content } = matter(file, MATTER_OPTIONS);
|
|
|
|
return {
|
|
props: {
|
|
frontmatter,
|
|
content,
|
|
navLinks,
|
|
lastModified: getParsedDate(lastModified.mtime, {
|
|
month: 'numeric',
|
|
day: 'numeric',
|
|
year: 'numeric'
|
|
})
|
|
}
|
|
};
|
|
};
|
|
|
|
interface Props {
|
|
frontmatter: {
|
|
[key: string]: string;
|
|
};
|
|
content: string;
|
|
navLinks: NavLink[];
|
|
lastModified: string;
|
|
}
|
|
|
|
const DocPage: NextPage<Props> = ({ frontmatter, content, navLinks, lastModified }) => {
|
|
const router = useRouter();
|
|
|
|
useEffect(() => {
|
|
const id = router.asPath.split('#')[1];
|
|
const element = document.getElementById(id);
|
|
|
|
if (!element) {
|
|
return;
|
|
}
|
|
|
|
element.scrollIntoView({ behavior: 'smooth', block: 'start' });
|
|
}, [router.asPath]);
|
|
|
|
return (
|
|
<>
|
|
<PageMetadata title={frontmatter.title} description={frontmatter.description} />
|
|
|
|
<main>
|
|
<Flex direction={{ base: 'column', lg: 'row' }} gap={{ base: 4, lg: 8 }}>
|
|
<Stack>
|
|
<DocsNav navLinks={navLinks} />
|
|
</Stack>
|
|
|
|
<Stack pb={4} width='100%'>
|
|
<Stack mb={16}>
|
|
<Breadcrumbs />
|
|
<Heading as='h1' mt='4 !important' mb={0} {...textStyles.header1}>
|
|
{frontmatter.title}
|
|
</Heading>
|
|
<Text as='span' mt='0 !important'>
|
|
last edited {lastModified}
|
|
</Text>
|
|
</Stack>
|
|
|
|
<Flex width='100%' placeContent='space-between'>
|
|
<Stack maxW='768px' sx={{ '*:first-child': { marginTop: '0 !important' } }}>
|
|
<ReactMarkdown
|
|
remarkPlugins={[gfm]}
|
|
rehypePlugins={[rehypeRaw]}
|
|
components={ChakraUIRenderer(MDComponents)}
|
|
>
|
|
{content}
|
|
</ReactMarkdown>
|
|
</Stack>
|
|
|
|
<Stack display={{ base: 'none', xl: 'block' }} w={48}>
|
|
<DocumentNav content={content} />
|
|
</Stack>
|
|
</Flex>
|
|
</Stack>
|
|
</Flex>
|
|
</main>
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default DocPage;
|