import { computed, ref, watch, defineComponent, provide } from 'vue'; import Checkbox from './Checkbox'; import { useInjectFormItemContext } from '../form/FormItemContext'; import useConfigInject from '../_util/hooks/useConfigInject'; import type { CheckboxOptionType } from './interface'; import { CheckboxGroupContextKey, checkboxGroupProps } from './interface'; export default defineComponent({ name: 'ACheckboxGroup', props: checkboxGroupProps(), emits: ['change', 'update:value'], setup(props, { slots, emit, expose }) { const formItemContext = useInjectFormItemContext(); const { prefixCls, direction } = useConfigInject('checkbox', props); const mergedValue = ref((props.value === undefined ? props.defaultValue : props.value) || []); watch( () => props.value, () => { mergedValue.value = props.value || []; }, ); const options = computed(() => { return props.options.map(option => { if (typeof option === 'string') { return { label: option, value: option, }; } return option; }); }); const triggerUpdate = ref(Symbol()); const registeredValuesMap = ref>(new Map()); const cancelValue = (id: Symbol) => { registeredValuesMap.value.delete(id); triggerUpdate.value = Symbol(); }; const registerValue = (id: Symbol, value: string) => { registeredValuesMap.value.set(id, value); triggerUpdate.value = Symbol(); }; const registeredValues = ref(new Map()); watch(triggerUpdate, () => { const valuseMap = new Map(); for (const value of registeredValuesMap.value.values()) { valuseMap.set(value, true); } registeredValues.value = valuseMap; }); const toggleOption = (option: CheckboxOptionType) => { const optionIndex = mergedValue.value.indexOf(option.value); const value = [...mergedValue.value]; if (optionIndex === -1) { value.push(option.value); } else { value.splice(optionIndex, 1); } if (props.value === undefined) { mergedValue.value = value; } const val = value .filter(val => registeredValues.value.has(val)) .sort((a, b) => { const indexA = options.value.findIndex(opt => opt.value === a); const indexB = options.value.findIndex(opt => opt.value === b); return indexA - indexB; }); emit('update:value', val); emit('change', val); formItemContext.onFieldChange(); }; provide(CheckboxGroupContextKey, { cancelValue, registerValue, toggleOption, mergedValue, name: computed(() => props.name), disabled: computed(() => props.disabled), }); expose({ mergedValue, }); return () => { const { id = formItemContext.id.value } = props; let children = null; const groupPrefixCls = `${prefixCls.value}-group`; if (options.value && options.value.length > 0) { children = options.value.map(option => ( {option.label === undefined ? slots.label?.(option) : option.label} )); } return (
{children || slots.default?.()}
); }; }, });