diff --git a/packages/ui/src/components/checkbox/checkbox.primitive.tsx b/packages/ui/src/components/checkbox/checkbox.primitive.tsx new file mode 100644 index 0000000..ebb57fc --- /dev/null +++ b/packages/ui/src/components/checkbox/checkbox.primitive.tsx @@ -0,0 +1,212 @@ +import { useEffect, useRef, useState } from 'react'; +import { + useComposeRefs, + useControllableState, + usePrevious, + useSize, +} from '../../hooks'; +import { + type HTMLFavolinkProps, + type HTMLFavolinkPropsWithout, + type HTMLProps, + createContext, + favolink, + forwardRef, +} from '../../system'; +import { mergeFns, mergeStyles } from '../../utils'; + +type CheckboxContextValue = { + state: boolean; + disabled?: boolean; +}; + +const [CheckboxProvider, useCheckboxContext] = + createContext({ + name: 'CheckboxContext', + hookName: 'useCheckboxContext', + providerName: '', + }); + +type CheckboxProps = HTMLFavolinkPropsWithout< + 'button', + 'checked' | 'defaultChecked' +> & { + checked?: boolean; + defaultChecked?: boolean; + required?: boolean; + onCheckedChange?: (state: boolean) => void; +}; + +const Checkbox = forwardRef( + function Checkbox(props, forwardedRef) { + const { + name, + value = 'on', + required, + disabled, + defaultChecked, + checked: checkedProp, + onCheckedChange, + ...restProps + } = props; + + const [button, setButton] = useState(null); + const [checked = false, setChecked] = useControllableState({ + value: checkedProp, + defaultValue: defaultChecked, + onChange: onCheckedChange, + }); + + const composedRefs = useComposeRefs(forwardedRef, (button) => { + setButton(button); + }); + const initialCheckedStateRef = useRef(checked); + const hasConsumerStoppedPropagation = useRef(false); + + const isFormControl = button ? Boolean(button.closest('form')) : true; + + useEffect(() => { + const form = button?.form; + + if (!form) { + return; + } + + function reset() { + setChecked(initialCheckedStateRef.current); + } + + form.addEventListener('reset', reset); + + return () => { + form.removeEventListener('reset', reset); + }; + }, [button, setChecked]); + + return ( + + { + if (event.key === 'Enter') { + event.preventDefault(); + } + })} + onClick={mergeFns(props.onClick, (event) => { + setChecked(!checked); + + if (isFormControl) { + hasConsumerStoppedPropagation.current = + event.isPropagationStopped(); + + if (!hasConsumerStoppedPropagation.current) { + event.stopPropagation(); + } + } + })} + /> + {isFormControl && ( + + )} + + ); + }, +); + +Checkbox.displayName = 'Checkbox'; + +type CheckboxIndicatorProps = HTMLFavolinkProps<'span'>; + +const CheckboxIndicator = forwardRef( + function CheckboxIndicator(props, forwardedRef) { + const context = useCheckboxContext(); + + return ( + context.state && ( + + ) + ); + }, +); + +CheckboxIndicator.displayName = 'CheckboxIndicator'; + +type BubbleInputProps = HTMLProps<'input'> & { + bubbles: boolean; + control: HTMLElement | null; +}; + +function BubbleInput(props: BubbleInputProps) { + const { control, checked, bubbles = true, ...restProps } = props; + + const ref = useRef(null); + const controlSize = useSize(control); + const prevChecked = usePrevious(checked); + + useEffect(() => { + const input = ref.current; + const inputPrototype = window.HTMLInputElement.prototype; + const checkedDescriptor = Object.getOwnPropertyDescriptor( + inputPrototype, + 'checked', + ) as PropertyDescriptor; + const setChecked = checkedDescriptor.set?.bind(input); + + if (prevChecked === checked || !setChecked) { + return; + } + + const event = new Event('click', { bubbles }); + + setChecked.call(input, checked); + input?.dispatchEvent(event); + }, [bubbles, prevChecked, checked]); + + return ( + + ); +} + +export { + Checkbox as Root, + type CheckboxProps as RootProps, + CheckboxIndicator as Indicator, + type CheckboxIndicatorProps as IndicatorProps, +}; diff --git a/packages/ui/src/components/checkbox/index.ts b/packages/ui/src/components/checkbox/index.ts new file mode 100644 index 0000000..24486f6 --- /dev/null +++ b/packages/ui/src/components/checkbox/index.ts @@ -0,0 +1,2 @@ +/* eslint-disable @stylistic/padding-line-between-statements, react-refresh/only-export-components */ +export * as CheckboxPrimitive from './checkbox.primitive';