You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
85 lines
2.1 KiB
85 lines
2.1 KiB
import { Fragment, ReactNode } from "react"; |
|
import { |
|
Transition as HeadlessTransition, |
|
TransitionClasses, |
|
} from "@headlessui/react"; |
|
|
|
type TransitionAnimations = "slide-down" | "slide-up" | "fade" | "none"; |
|
|
|
interface Props { |
|
show?: boolean; |
|
durationClass?: string; |
|
animation: TransitionAnimations; |
|
className?: string; |
|
children?: ReactNode; |
|
appearOnMount?: boolean; |
|
isChild?: boolean; |
|
} |
|
|
|
function getClasses( |
|
animation: TransitionAnimations, |
|
duration: string |
|
): TransitionClasses { |
|
if (animation === "slide-down") { |
|
return { |
|
leave: `transition-[transform,opacity] ${duration}`, |
|
leaveFrom: "opacity-100 translate-y-0", |
|
leaveTo: "-translate-y-4 opacity-0", |
|
enter: `transition-[transform,opacity] ${duration}`, |
|
enterFrom: "opacity-0 -translate-y-4", |
|
enterTo: "translate-y-0 opacity-100", |
|
}; |
|
} |
|
|
|
if (animation === "slide-up") { |
|
return { |
|
leave: `transition-[transform,opacity] ${duration}`, |
|
leaveFrom: "opacity-100 translate-y-0", |
|
leaveTo: "translate-y-4 opacity-0", |
|
enter: `transition-[transform,opacity] ${duration}`, |
|
enterFrom: "opacity-0 translate-y-4", |
|
enterTo: "translate-y-0 opacity-100", |
|
}; |
|
} |
|
|
|
if (animation === "fade") { |
|
return { |
|
leave: `transition-[transform,opacity] ${duration}`, |
|
leaveFrom: "opacity-100", |
|
leaveTo: "opacity-0", |
|
enter: `transition-[transform,opacity] ${duration}`, |
|
enterFrom: "opacity-0", |
|
enterTo: "opacity-100", |
|
}; |
|
} |
|
|
|
return {}; |
|
} |
|
|
|
export function Transition(props: Props) { |
|
const duration = props.durationClass ?? "duration-200"; |
|
const classes = getClasses(props.animation, duration); |
|
|
|
if (props.isChild) { |
|
return ( |
|
<HeadlessTransition.Child |
|
as={Fragment} |
|
appear={props.appearOnMount} |
|
{...classes} |
|
> |
|
<div className={props.className}>{props.children}</div> |
|
</HeadlessTransition.Child> |
|
); |
|
} |
|
|
|
return ( |
|
<HeadlessTransition |
|
show={props.show} |
|
as={Fragment} |
|
appear={props.appearOnMount} |
|
{...classes} |
|
> |
|
<div className={props.className}>{props.children}</div> |
|
</HeadlessTransition> |
|
); |
|
}
|
|
|