117 lines
3.8 KiB
Vue
117 lines
3.8 KiB
Vue
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<Map<Symbol, string>>(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 => (
|
|
<Checkbox
|
|
prefixCls={prefixCls.value}
|
|
key={option.value.toString()}
|
|
disabled={'disabled' in option ? option.disabled : props.disabled}
|
|
indeterminate={option.indeterminate}
|
|
value={option.value}
|
|
checked={mergedValue.value.indexOf(option.value) !== -1}
|
|
onChange={option.onChange}
|
|
class={`${groupPrefixCls}-item`}
|
|
>
|
|
{option.label === undefined ? slots.label?.(option) : option.label}
|
|
</Checkbox>
|
|
));
|
|
}
|
|
return (
|
|
<div
|
|
class={[groupPrefixCls, { [`${groupPrefixCls}-rtl`]: direction.value === 'rtl' }]}
|
|
id={id}
|
|
>
|
|
{children || slots.default?.()}
|
|
</div>
|
|
);
|
|
};
|
|
},
|
|
});
|