Updated the import statements in multiple component files to use the updated package name for shared resources. This change was necessary for import consistency and to ensure proper functioning of the components as per the new package name "@pswui-lib/shared@1.0.0".
114 lines
4.2 KiB
TypeScript
114 lines
4.2 KiB
TypeScript
import React from "react";
|
|
import { VariantProps, vcn } from "@pswui-lib/shared@1.0.0";
|
|
|
|
const checkboxColors = {
|
|
background: {
|
|
default: "bg-neutral-200 dark:bg-neutral-700",
|
|
hover: "hover:bg-neutral-300 dark:hover:bg-neutral-600",
|
|
checked:
|
|
"has-[input[type=checkbox]:checked]:bg-neutral-300 dark:has-[input[type=checkbox]:checked]:bg-neutral-400",
|
|
checkedHover:
|
|
"has-[input[type=checkbox]:checked]:hover:bg-neutral-400 dark:has-[input[type=checkbox]:checked]:hover:bg-neutral-300",
|
|
disabled:
|
|
'has-[input[type="checkbox"]:disabled]:bg-neutral-100 dark:has-[input[type="checkbox"]:disabled]:bg-neutral-800',
|
|
disabledHover:
|
|
"has-[input[type='checkbox']:disabled]:hover:bg-neutral-100 dark:has-[input[type='checkbox']:disabled]:hover:bg-neutral-800",
|
|
disabledChecked:
|
|
"has-[input[type='checkbox']:disabled:checked]:bg-neutral-300 dark:has-[input[type='checkbox']:disabled:checked]:bg-neutral-700",
|
|
disabledCheckedHover:
|
|
"has-[input[type='checkbox']:disabled:checked]:hover:bg-neutral-300 dark:has-[input[type='checkbox']:disabled:checked]:hover:bg-neutral-700",
|
|
},
|
|
checkmark:
|
|
"text-black dark:text-white has-[input[type=checkbox]:disabled]:text-neutral-400 dark:has-[input[type=checkbox]:disabled]:text-neutral-500",
|
|
};
|
|
|
|
const [checkboxVariant, resolveCheckboxVariantProps] = vcn({
|
|
base: `inline-block rounded-md ${checkboxColors.checkmark} ${checkboxColors.background.disabled} ${checkboxColors.background.default} ${checkboxColors.background.hover} ${checkboxColors.background.checked} ${checkboxColors.background.checkedHover} ${checkboxColors.background.disabledChecked} ${checkboxColors.background.disabledCheckedHover} has-[input[type="checkbox"]:disabled]:cursor-not-allowed transition-colors duration-75 ease-in-out`,
|
|
variants: {
|
|
size: {
|
|
base: "size-[1em] p-0 [&>svg]:size-[1em]",
|
|
md: "size-[1.5em] p-0.5 [&>svg]:size-[1.25em]",
|
|
lg: "size-[1.75em] p-1 [&>svg]:size-[1.25em]",
|
|
},
|
|
},
|
|
defaults: {
|
|
size: "md",
|
|
},
|
|
});
|
|
|
|
interface CheckboxProps
|
|
extends VariantProps<typeof checkboxVariant>,
|
|
Omit<
|
|
React.ComponentPropsWithoutRef<"input">,
|
|
"type" | "className" | "size"
|
|
> {}
|
|
|
|
const Checkbox = React.forwardRef<HTMLInputElement, CheckboxProps>(
|
|
(props, ref) => {
|
|
const [variantProps, otherPropsCompressed] =
|
|
resolveCheckboxVariantProps(props);
|
|
const {
|
|
defaultChecked,
|
|
checked: propChecked,
|
|
onChange,
|
|
...otherPropsExtracted
|
|
} = otherPropsCompressed;
|
|
|
|
// internally handles checked, so we can use checked value without prop
|
|
const [checked, setChecked] = React.useState(defaultChecked ?? false);
|
|
React.useEffect(() => {
|
|
if (typeof propChecked === "boolean") {
|
|
setChecked(propChecked);
|
|
}
|
|
}, [propChecked]);
|
|
|
|
const internalRef = React.useRef<HTMLInputElement | null>(null);
|
|
|
|
return (
|
|
<>
|
|
<label className={checkboxVariant(variantProps)}>
|
|
<input
|
|
{...otherPropsExtracted}
|
|
defaultChecked={defaultChecked}
|
|
checked={
|
|
typeof defaultChecked === "boolean"
|
|
? undefined
|
|
: checked /* should be either uncontrolled (defaultChecked set) or controlled (checked set) */
|
|
}
|
|
onChange={(e) => {
|
|
setChecked(e.currentTarget.checked);
|
|
if (onChange) {
|
|
onChange(e);
|
|
}
|
|
}}
|
|
type="checkbox"
|
|
className="hidden"
|
|
ref={(el) => {
|
|
internalRef.current = el;
|
|
if (typeof ref === "function") {
|
|
ref(el);
|
|
} else if (ref) {
|
|
ref.current = el;
|
|
}
|
|
}}
|
|
/>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
width="1em"
|
|
height="1em"
|
|
viewBox="0 0 24 24"
|
|
className={`${checked ? "opacity-100" : "opacity-0"} transition-opacity duration-75 ease-in-out`}
|
|
>
|
|
<path
|
|
fill="currentColor"
|
|
d="M21 7L9 19l-5.5-5.5l1.41-1.41L9 16.17L19.59 5.59z"
|
|
></path>
|
|
</svg>
|
|
</label>
|
|
</>
|
|
);
|
|
},
|
|
);
|
|
|
|
export { Checkbox };
|