| import { useEffect, useRef } from "react"; |
| import Prism from "prismjs"; |
| import "prismjs/themes/prism-tomorrow.css"; |
| import "prismjs/components/prism-bash"; |
| import "prismjs/components/prism-javascript"; |
| import "prismjs/components/prism-typescript"; |
| import "prismjs/components/prism-json"; |
| import { Button } from "@/components/ui/button"; |
| import { Copy, Check } from "lucide-react"; |
| import { useState } from "react"; |
|
|
| interface CodeBlockProps { |
| code: string; |
| language: "bash" | "javascript" | "typescript" | "json"; |
| showCopy?: boolean; |
| } |
|
|
| export function CodeBlock({ code, language, showCopy = true }: CodeBlockProps) { |
| const codeRef = useRef<HTMLElement>(null); |
| const [copied, setCopied] = useState(false); |
|
|
| useEffect(() => { |
| if (codeRef.current) { |
| Prism.highlightElement(codeRef.current); |
| } |
| }, [code, language]); |
|
|
| const handleCopy = () => { |
| navigator.clipboard.writeText(code); |
| setCopied(true); |
| setTimeout(() => setCopied(false), 2000); |
| }; |
|
|
| return ( |
| <div className="relative group"> |
| {showCopy && ( |
| <Button |
| variant="ghost" |
| size="sm" |
| onClick={handleCopy} |
| className="absolute top-2 right-2 h-8 opacity-0 group-hover:opacity-100 transition-opacity bg-white/10 hover:bg-white/20 z-10" |
| > |
| {copied ? ( |
| <> |
| <Check className="w-3 h-3 mr-1" /> |
| Copied |
| </> |
| ) : ( |
| <> |
| <Copy className="w-3 h-3 mr-1" /> |
| Copy |
| </> |
| )} |
| </Button> |
| )} |
| <pre className="!bg-[#1e1e1e] !border !border-white/10 !rounded-lg !p-4 !m-0 overflow-x-auto"> |
| <code ref={codeRef} className={`language-${language} !text-sm`}> |
| {code} |
| </code> |
| </pre> |
| </div> |
| ); |
| } |