import * as React from 'react'; import * as PropTypes from 'prop-types'; import { polyfill } from 'react-lifecycles-compat'; import classNames from 'classnames'; import shallowEqual from 'shallowequal'; import omit from 'omit.js'; import Checkbox from './Checkbox'; export type CheckboxValueType = string | number | boolean; export interface CheckboxOptionType { label: string; value: CheckboxValueType; disabled?: boolean; } export interface AbstractCheckboxGroupProps { prefixCls?: string; className?: string; options?: Array; disabled?: boolean; style?: React.CSSProperties; } export interface CheckboxGroupProps extends AbstractCheckboxGroupProps { defaultValue?: Array; value?: Array; onChange?: (checkedValue: Array) => void; } export interface CheckboxGroupState { value: any; } export interface CheckboxGroupContext { checkboxGroup: { toggleOption: (option: CheckboxOptionType) => void; value: any; disabled: boolean; }; } class CheckboxGroup extends React.Component { static defaultProps = { options: [], prefixCls: 'ant-checkbox', }; static propTypes = { defaultValue: PropTypes.array, value: PropTypes.array, options: PropTypes.array.isRequired, onChange: PropTypes.func, }; static childContextTypes = { checkboxGroup: PropTypes.any, }; static getDerivedStateFromProps(nextProps: CheckboxGroupProps) { if ('value' in nextProps) { return { value: nextProps.value || [], }; } return null; } constructor(props: CheckboxGroupProps) { super(props); this.state = { value: props.value || props.defaultValue || [], }; } getChildContext() { return { checkboxGroup: { toggleOption: this.toggleOption, value: this.state.value, disabled: this.props.disabled, }, }; } shouldComponentUpdate(nextProps: CheckboxGroupProps, nextState: CheckboxGroupState) { return !shallowEqual(this.props, nextProps) || !shallowEqual(this.state, nextState); } getOptions() { const { options } = this.props; // https://github.com/Microsoft/TypeScript/issues/7960 return (options as Array).map(option => { if (typeof option === 'string') { return { label: option, value: option, } as CheckboxOptionType; } return option; }); } toggleOption = (option: CheckboxOptionType) => { const optionIndex = this.state.value.indexOf(option.value); const value = [...this.state.value]; if (optionIndex === - 1) { value.push(option.value); } else { value.splice(optionIndex, 1); } if (!('value' in this.props)) { this.setState({ value }); } const onChange = this.props.onChange; if (onChange) { onChange(value); } } render() { const { props, state } = this; const { prefixCls, className, style, options, ...restProps } = props; const groupPrefixCls = `${prefixCls}-group`; const domProps = omit(restProps, ['children', 'defaultValue', 'value', 'onChange', 'disabled']); let children = props.children; if (options && options.length > 0) { children = this.getOptions().map(option => ( this.toggleOption(option)} className={`${groupPrefixCls}-item`} > {option.label} )); } const classString = classNames(groupPrefixCls, className); return (
{children}
); } } polyfill(CheckboxGroup); export default CheckboxGroup;