mirror of
https://github.com/supabase/supabase.git
synced 2026-05-29 17:01:12 +08:00
84 lines
2.4 KiB
TypeScript
84 lines
2.4 KiB
TypeScript
// [Terry]
|
|
// Delete this after we've implemented GuidesTableofContents and moved all guides
|
|
// and rename GuidesTableofContents to TableOfContents
|
|
|
|
import { FC } from 'react'
|
|
import { getAnchor, removeAnchor } from './CustomHTMLElements/CustomHTMLElements.utils'
|
|
|
|
interface TOCHeader {
|
|
id: number
|
|
lvl: number
|
|
seen: number
|
|
content: string
|
|
slug: string
|
|
}
|
|
|
|
interface Props {
|
|
toc: any
|
|
video?: string
|
|
}
|
|
|
|
const formatSlug = (slug: string) => {
|
|
// [Joshen] We will still provide support for headers declared like this:
|
|
// ## REST API {#rest-api-overview}
|
|
// At least for now, this was a docusaurus thing.
|
|
if (slug.includes('#')) return slug.split('-#')[1]
|
|
return slug
|
|
}
|
|
|
|
const formatTOCHeader = (content: string) => {
|
|
let begin = false
|
|
const res = []
|
|
for (const x of content) {
|
|
if (x === '`') {
|
|
if (!begin) {
|
|
begin = true
|
|
res.push(`<code class="text-xs border rounded bg-scale-400 border-scale-500">`)
|
|
} else {
|
|
begin = false
|
|
res.push(`</code>`)
|
|
}
|
|
} else {
|
|
res.push(x)
|
|
}
|
|
}
|
|
return res.join('')
|
|
}
|
|
|
|
const TableOfContents: FC<Props> = ({ toc, video }) => {
|
|
// [Joshen] markdown-toc doesn't seem to read maxdepth from the options passed in
|
|
// Our first level headers will be H2s (H1 is ignored), and we only show up to H3
|
|
|
|
return (
|
|
<>
|
|
{video && (
|
|
<div className="video-container">
|
|
<iframe
|
|
src="https://www.youtube-nocookie.com/embed/0Fs96oZ4se0"
|
|
frameBorder="1"
|
|
allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture"
|
|
allowFullScreen
|
|
></iframe>
|
|
</div>
|
|
)}
|
|
<ul className="toc-menu list-none pl-4 text-[0.8rem] grid gap-2 mt-12">
|
|
{(toc.json as TOCHeader[])
|
|
.filter((item) => item.lvl !== 1 && item.lvl <= 3)
|
|
.map((item: any, i: number) => {
|
|
return (
|
|
<li key={i} id={item.lvl} style={{ marginLeft: `${(item.lvl - 2) * 1}rem` }}>
|
|
<a
|
|
href={`#${formatSlug(item.slug)}`}
|
|
className="text-scale-1000 hover:text-brand-900 transition-colors"
|
|
dangerouslySetInnerHTML={{ __html: formatTOCHeader(removeAnchor(item.content)) }}
|
|
/>
|
|
</li>
|
|
)
|
|
})}
|
|
</ul>
|
|
</>
|
|
)
|
|
}
|
|
|
|
export default TableOfContents
|