import { Button } from "@pswui/Button"; import { useToast } from "@pswui/Toast"; import { type Component, forwardRef, useEffect, useMemo, useState, } from "react"; import { PrismLight, type SyntaxHighlighterProps, } from "react-syntax-highlighter"; import { duotoneSpace } from "react-syntax-highlighter/dist/cjs/styles/prism"; import css from "react-syntax-highlighter/dist/esm/languages/prism/css"; import js from "react-syntax-highlighter/dist/esm/languages/prism/javascript"; import jsx from "react-syntax-highlighter/dist/esm/languages/prism/jsx"; import markup from "react-syntax-highlighter/dist/esm/languages/prism/markup"; import tsx from "react-syntax-highlighter/dist/esm/languages/prism/tsx"; import ts from "react-syntax-highlighter/dist/esm/languages/prism/typescript"; import { twMerge } from "tailwind-merge"; const SyntaxHighlighter = PrismLight as unknown as typeof Component & { registerLanguage(name: string, func: T): void; alias(name: string, alias: string | string[]): void; alias(aliases: Record): void; }; SyntaxHighlighter.registerLanguage("javascript", js); SyntaxHighlighter.registerLanguage("typescript", ts); SyntaxHighlighter.registerLanguage("tsx", tsx); SyntaxHighlighter.registerLanguage("jsx", jsx); SyntaxHighlighter.registerLanguage("markup", markup); SyntaxHighlighter.registerLanguage("css", css); export const GITHUB_UI = "https://raw.githubusercontent.com/pswui/ui/main"; export const GITHUB_DOCS = "https://raw.githubusercontent.com/pswui/docs/main"; export const GITHUB_COMP = (componentName: string) => `${GITHUB_UI}/packages/react/components/${componentName}.tsx`; export const GITHUB_DIR_COMP = (componentName: string, source: string) => `${GITHUB_UI}/packages/react/components/${componentName}/${source}`; export const GITHUB_COMP_PREVIEW = (componentName: string) => `${GITHUB_DOCS}/src/docs/components/${componentName}Blocks/Preview.tsx`; export const GITHUB_STORY = (componentName: string, storyName: string) => `${GITHUB_DOCS}/src/docs/components/${componentName}Blocks/Examples/${storyName}.tsx`; export type TEMPLATE = Record< string, Record >; const TEMPLATE_REMOVE_REGEX = /\/\*\s*remove\s*\*\/(.|\n)*?\/\*\s*end\s*\*\//g; const TEMPLATE_REPLACE_REGEX = /\/\*\s*replace\s*\*\/(.|\n)*?\/\*\s*with\s*\n((.|\n)+)\n\s*\*\//g; export const LoadedCode = forwardRef< HTMLDivElement, { from: string; className?: string; template?: TEMPLATE } >(({ from, className, template }, ref) => { const [state, setState] = useState(); const { toast } = useToast(); useEffect(() => { (async () => { const res = await fetch(from); const text = await res.text(); setState(text); })(); }, [from]); const postProcessedCode = useMemo(() => { if (!state) return ""; if (!template) return state; let templatedCode = state; templatedCode = templatedCode .replaceAll(TEMPLATE_REMOVE_REGEX, "") .replaceAll(TEMPLATE_REPLACE_REGEX, "$2"); for (const [componentName, componentTemplateProps] of Object.entries( template, )) { for (const [propName, propValue] of Object.entries( componentTemplateProps, )) { const regex = new RegExp( `(<${componentName.slice(0, componentName.length - 5)}\\b[^>]*?)(\n?\\s+)${propName}={${componentName}.${propName}}`, ); templatedCode = templatedCode.replace( regex, typeof propValue === "undefined" ? "$1" : typeof propValue === "string" ? `\$1$2 ${propName}="${propValue}"` : `$1$2 ${propName}={${propValue}}`, ); } } return templatedCode; }, [state, template]); return (
{postProcessedCode}
); }); LoadedCode.displayName = "LoadedCode"; export const Code = forwardRef< HTMLDivElement, { children: string; className?: string; language: string } >(({ children, className, language }, ref) => { const { toast } = useToast(); return (
{children}
); });