forked from baron/baron-sso
69 lines
2.1 KiB
TypeScript
69 lines
2.1 KiB
TypeScript
import * as React from "react";
|
|
import { cn } from "../../lib/utils";
|
|
|
|
interface SwitchProps
|
|
extends Omit<React.ButtonHTMLAttributes<HTMLButtonElement>, "onChange"> {
|
|
checked?: boolean;
|
|
defaultChecked?: boolean;
|
|
onCheckedChange?: (checked: boolean) => void;
|
|
}
|
|
|
|
const Switch = React.forwardRef<HTMLButtonElement, SwitchProps>(
|
|
(
|
|
{
|
|
className,
|
|
checked,
|
|
defaultChecked = false,
|
|
disabled,
|
|
onCheckedChange,
|
|
onClick,
|
|
...props
|
|
},
|
|
ref,
|
|
) => {
|
|
const isControlled = checked !== undefined;
|
|
const [internalChecked, setInternalChecked] =
|
|
React.useState(defaultChecked);
|
|
const currentChecked = isControlled ? checked : internalChecked;
|
|
|
|
const handleClick = (event: React.MouseEvent<HTMLButtonElement>) => {
|
|
onClick?.(event);
|
|
if (event.defaultPrevented || disabled) {
|
|
return;
|
|
}
|
|
const nextChecked = !currentChecked;
|
|
if (!isControlled) {
|
|
setInternalChecked(nextChecked);
|
|
}
|
|
onCheckedChange?.(nextChecked);
|
|
};
|
|
|
|
return (
|
|
<button
|
|
type="button"
|
|
role="switch"
|
|
aria-checked={currentChecked}
|
|
data-state={currentChecked ? "checked" : "unchecked"}
|
|
className={cn(
|
|
"peer inline-flex h-5 w-10 shrink-0 cursor-pointer items-center rounded-full border-2 border-transparent bg-input transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background disabled:cursor-not-allowed disabled:opacity-50 data-[state=checked]:bg-primary data-[state=unchecked]:bg-muted/50",
|
|
className,
|
|
)}
|
|
disabled={disabled}
|
|
onClick={handleClick}
|
|
ref={ref}
|
|
{...props}
|
|
>
|
|
<span
|
|
data-state={currentChecked ? "checked" : "unchecked"}
|
|
className={cn(
|
|
"pointer-events-none block h-4 w-4 rounded-full bg-background shadow-lg ring-0 transition-transform data-[state=checked]:translate-x-4 data-[state=unchecked]:translate-x-0",
|
|
)}
|
|
/>
|
|
</button>
|
|
);
|
|
},
|
|
);
|
|
Switch.displayName = "Switch";
|
|
|
|
export { Switch };
|