diff --git a/packages/react/src/App.tsx b/packages/react/src/App.tsx index 288aed2..71b0d02 100644 --- a/packages/react/src/App.tsx +++ b/packages/react/src/App.tsx @@ -27,7 +27,66 @@ import ComponentsCheckbox, { import ComponentsDialog, { tableOfContents as componentsDialogToc, } from "./docs/components/Dialog.mdx"; -import { forwardRef } from "react"; +import { ForwardedRef, forwardRef, useContext, useEffect, useRef } from "react"; +import { HeadingContext } from "./HeadingContext"; + +function buildThresholdList() { + let thresholds = []; + let numSteps = 20; + + for (let i = 1.0; i <= numSteps; i++) { + let ratio = i / numSteps; + thresholds.push(ratio); + } + + thresholds.push(0); + return thresholds; +} + +function HashedHeaders(Level: `h${1 | 2 | 3 | 4 | 5 | 6}`) { + return (prop: any, ref: ForwardedRef) => { + const internalRef = useRef(null); + const [_, setActiveHeadings] = useContext(HeadingContext); + + useEffect(() => { + const observer = new IntersectionObserver( + ([{ target, intersectionRatio }]) => { + if (intersectionRatio > 0.5) { + setActiveHeadings((prev) => [...prev, target.id]); + } else { + setActiveHeadings((prev) => prev.filter((id) => id !== target.id)); + } + }, + { + root: null, + rootMargin: "0px", + threshold: buildThresholdList(), + } + ); + if (internalRef.current) { + observer.observe(internalRef.current); + } + return () => { + observer.disconnect(); + }; + }, [internalRef.current]); + + return ( + { + internalRef.current = el; + if (typeof ref === "function") { + ref(el); + } else if (el && ref) { + ref.current = el; + } + }} + /> + ); + }; +} const overrideComponents = { pre: forwardRef((props: any, ref) => ( @@ -45,6 +104,12 @@ const overrideComponents = { )), + h1: forwardRef(HashedHeaders("h1")), + h2: forwardRef(HashedHeaders("h2")), + h3: forwardRef(HashedHeaders("h3")), + h4: forwardRef(HashedHeaders("h4")), + h5: forwardRef(HashedHeaders("h5")), + h6: forwardRef(HashedHeaders("h6")), }; const router = createBrowserRouter( diff --git a/packages/react/src/DynamicLayout.tsx b/packages/react/src/DynamicLayout.tsx index 88542c3..e5f1fbd 100644 --- a/packages/react/src/DynamicLayout.tsx +++ b/packages/react/src/DynamicLayout.tsx @@ -1,9 +1,11 @@ -import { ReactNode, Fragment } from "react"; +import { ReactNode, Fragment, useState, useContext } from "react"; import { type Toc } from "@stefanprobst/rehype-extract-toc"; import { useLocation } from "react-router-dom"; +import { HeadingContext } from "./HeadingContext"; function RecursivelyToc({ toc }: { toc: Toc }) { const location = useLocation(); + const [activeHeadings] = useContext(HeadingContext); return (
    @@ -16,7 +18,9 @@ function RecursivelyToc({ toc }: { toc: Toc }) { className="text-neutral-500 data-[active='true']:text-black dark:data-[active='true']:text-white text-sm font-medium" style={{ paddingLeft: `${tocEntry.depth - 1}rem` }} data-active={ - location.hash.length > 0 + activeHeadings.includes(tocEntry.id ?? "") + ? true + : location.hash.length > 0 ? location.hash === `#${tocEntry.id}` : true } @@ -40,8 +44,10 @@ export default function DynamicLayout({ children: ReactNode; toc: Toc; }) { + const [activeHeadings, setActiveHeadings] = useState([]); + return ( - <> +
    {children} @@ -52,6 +58,6 @@ export default function DynamicLayout({ - + ); } diff --git a/packages/react/src/HeadingContext.tsx b/packages/react/src/HeadingContext.tsx new file mode 100644 index 0000000..4ce18e2 --- /dev/null +++ b/packages/react/src/HeadingContext.tsx @@ -0,0 +1,12 @@ +import { Dispatch, SetStateAction, createContext } from "react"; + +export const HeadingContext = createContext< + [string[], Dispatch>] +>([ + [], + () => { + if (process.env && process.env.NODE_ENV === "development") { + console.log("HeadingContext outside"); + } + }, +]);