diff --git a/packages/react/src/App.tsx b/packages/react/src/App.tsx index 1717166..b559bda 100644 --- a/packages/react/src/App.tsx +++ b/packages/react/src/App.tsx @@ -29,7 +29,9 @@ import React, { useContext, useEffect, useRef, + useState, } from "react"; +import { Tooltip, TooltipContent } from "@components/Tooltip.tsx"; function buildThresholdList() { const thresholds: number[] = []; @@ -49,6 +51,8 @@ function HashedHeaders(Level: `h${1 | 2 | 3 | 4 | 5 | 6}`) { const internalRef = useRef(null); const [_, setActiveHeadings] = useContext(HeadingContext); + const { children, ...restProp } = prop; + useEffect(() => { const observer = new IntersectionObserver( ([{ target, intersectionRatio }]) => { @@ -72,19 +76,58 @@ function HashedHeaders(Level: `h${1 | 2 | 3 | 4 | 5 | 6}`) { }; }, [internalRef.current]); + const [status, setStatus] = useState<"normal" | "error" | "success">( + "normal", + ); + const messages = { + normal: "Click to copy link", + success: "Copied link!", + error: "Failed to copy..", + }; + + useEffect(() => { + if (status !== "normal") { + const timeout = setTimeout(() => { + setStatus("normal"); + }, 3000); + return () => { + clearTimeout(timeout); + }; + } + }, [status]); + return ( - { - internalRef.current = el; - if (typeof ref === "function") { - ref(el); - } else if (el && ref) { - ref.current = el; - } - }} - /> + + { + internalRef.current = el; + if (typeof ref === "function") { + ref(el); + } else if (el && ref) { + ref.current = el; + } + }} + className={`${prop.className} cursor-pointer select-none`} + onClick={async (e) => { + try { + await navigator.clipboard.writeText( + window.location.href.split("#")[0] + "#" + e.currentTarget.id, + ); + setStatus("success"); + } catch (e) { + setStatus("error"); + } + }} + {...restProp} + > + {children} + +

+ {messages[status]} +

+
+
+
); }; }