mirror of
https://github.com/supabase/supabase.git
synced 2026-05-08 01:40:13 -04:00
4a0bb36ca8
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: Ivan Vasilov <vasilov.ivan@gmail.com>
70 lines
1.7 KiB
TypeScript
70 lines
1.7 KiB
TypeScript
import { Check, Copy } from 'lucide-react'
|
|
import { ComponentProps, forwardRef, useEffect, useState } from 'react'
|
|
import { Button, cn, copyToClipboard } from 'ui'
|
|
|
|
type CopyButtonBaseProps = {
|
|
iconOnly?: boolean
|
|
copyLabel?: string
|
|
copiedLabel?: string
|
|
}
|
|
|
|
type CopyButtonWithText = CopyButtonBaseProps & {
|
|
text: string
|
|
asyncText?: never
|
|
}
|
|
|
|
type CopyButtonWithAsyncText = CopyButtonBaseProps & {
|
|
text?: never
|
|
asyncText: () => Promise<string> | string
|
|
}
|
|
|
|
export type CopyButtonProps = (CopyButtonWithText | CopyButtonWithAsyncText) &
|
|
ComponentProps<typeof Button>
|
|
|
|
const CopyButton = forwardRef<HTMLButtonElement, CopyButtonProps>(
|
|
(
|
|
{
|
|
text,
|
|
asyncText,
|
|
iconOnly = false,
|
|
children,
|
|
onClick,
|
|
copyLabel = 'Copy',
|
|
copiedLabel = 'Copied',
|
|
...props
|
|
},
|
|
ref
|
|
) => {
|
|
const [showCopied, setShowCopied] = useState(false)
|
|
|
|
useEffect(() => {
|
|
if (!showCopied) return
|
|
const timer = setTimeout(() => setShowCopied(false), 2000)
|
|
return () => clearTimeout(timer)
|
|
}, [showCopied])
|
|
|
|
return (
|
|
<Button
|
|
ref={ref}
|
|
onClick={(e) => {
|
|
const textToCopy = asyncText ? asyncText() : text
|
|
setShowCopied(true)
|
|
copyToClipboard(textToCopy)
|
|
onClick?.(e)
|
|
}}
|
|
{...props}
|
|
className={cn({ 'px-1': iconOnly }, props.className)}
|
|
icon={
|
|
showCopied ? <Check strokeWidth={2} className="text-brand" /> : (props.icon ?? <Copy />)
|
|
}
|
|
>
|
|
{!iconOnly && <>{children ?? (showCopied ? copiedLabel : copyLabel)}</>}
|
|
</Button>
|
|
)
|
|
}
|
|
)
|
|
|
|
CopyButton.displayName = 'CopyButton'
|
|
|
|
export default CopyButton
|