import * as React from 'react'; import * as PropTypes from 'prop-types'; import classNames from 'classnames'; import List, { TransferListProps } from './list'; import Operation from './operation'; import Search from './search'; import warning from '../_util/warning'; import LocaleReceiver from '../locale-provider/LocaleReceiver'; import defaultLocale from '../locale-provider/default'; import { ConfigConsumer, ConfigConsumerProps, RenderEmptyHandler } from '../config-provider'; import { polyfill } from 'react-lifecycles-compat'; import { TransferListBodyProps } from './renderListBody'; export { TransferListProps } from './list'; export { TransferOperationProps } from './operation'; export { TransferSearchProps } from './search'; export type TransferDirection = 'left' | 'right'; export interface RenderResultObject { label: React.ReactElement; value: string; } export type RenderResult = React.ReactElement | RenderResultObject | string | null; type TransferRender = (item: TransferItem) => RenderResult; export interface TransferItem { key: string; title: string; description?: string; disabled?: boolean; [name: string]: any; } export interface TransferProps { prefixCls?: string; className?: string; disabled?: boolean; dataSource: TransferItem[]; targetKeys?: string[]; selectedKeys?: string[]; render?: TransferRender; onChange?: (targetKeys: string[], direction: string, moveKeys: string[]) => void; onSelectChange?: (sourceSelectedKeys: string[], targetSelectedKeys: string[]) => void; style?: React.CSSProperties; listStyle?: React.CSSProperties; operationStyle?: React.CSSProperties; titles?: string[]; operations?: string[]; showSearch?: boolean; filterOption?: (inputValue: string, item: TransferItem) => boolean; searchPlaceholder?: string; notFoundContent?: React.ReactNode; locale?: {}; footer?: (props: TransferListProps) => React.ReactNode; body?: (props: TransferListProps) => React.ReactNode; rowKey?: (record: TransferItem) => string; onSearchChange?: (direction: TransferDirection, e: React.ChangeEvent) => void; onSearch?: (direction: TransferDirection, value: string) => void; lazy?: {} | boolean; onScroll?: (direction: TransferDirection, e: React.SyntheticEvent) => void; children?: (props: TransferListBodyProps) => React.ReactNode; showSelectAll?: boolean; } export interface TransferLocale { titles: string[]; notFoundContent: string; searchPlaceholder: string; itemUnit: string; itemsUnit: string; } class Transfer extends React.Component { // For high-level customized Transfer @dqaria static List = List; static Operation = Operation; static Search = Search; static defaultProps = { dataSource: [], locale: {}, showSearch: false, }; static propTypes = { prefixCls: PropTypes.string, disabled: PropTypes.bool, dataSource: PropTypes.array as PropTypes.Validator, render: PropTypes.func, targetKeys: PropTypes.array, onChange: PropTypes.func, height: PropTypes.number, style: PropTypes.object, listStyle: PropTypes.object, operationStyle: PropTypes.object, className: PropTypes.string, titles: PropTypes.array, operations: PropTypes.array, showSearch: PropTypes.bool, filterOption: PropTypes.func, searchPlaceholder: PropTypes.string, notFoundContent: PropTypes.node, locale: PropTypes.object, body: PropTypes.func, footer: PropTypes.func, rowKey: PropTypes.func, lazy: PropTypes.oneOfType([PropTypes.object, PropTypes.bool]), }; static getDerivedStateFromProps(nextProps: TransferProps) { if (nextProps.selectedKeys) { const targetKeys = nextProps.targetKeys || []; return { sourceSelectedKeys: nextProps.selectedKeys.filter(key => !targetKeys.includes(key)), targetSelectedKeys: nextProps.selectedKeys.filter(key => targetKeys.includes(key)), }; } return null; } separatedDataSource: { leftDataSource: TransferItem[]; rightDataSource: TransferItem[]; } | null = null; constructor(props: TransferProps) { super(props); warning( !('notFoundContent' in props || 'searchPlaceholder' in props), 'Transfer', '`notFoundContent` and `searchPlaceholder` will be removed, ' + 'please use `locale` instead.', ); warning( !('body' in props), 'Transfer', '`body` is internal usage and will bre removed, please use `children` instead.', ); const { selectedKeys = [], targetKeys = [] } = props; this.state = { sourceSelectedKeys: selectedKeys.filter(key => targetKeys.indexOf(key) === -1), targetSelectedKeys: selectedKeys.filter(key => targetKeys.indexOf(key) > -1), }; } separateDataSource(props: TransferProps) { const { dataSource, rowKey, targetKeys = [] } = props; const leftDataSource: TransferItem[] = []; const rightDataSource: TransferItem[] = new Array(targetKeys.length); dataSource.forEach(record => { if (rowKey) { record.key = rowKey(record); } // rightDataSource should be ordered by targetKeys // leftDataSource should be ordered by dataSource const indexOfKey = targetKeys.indexOf(record.key); if (indexOfKey !== -1) { rightDataSource[indexOfKey] = record; } else { leftDataSource.push(record); } }); return { leftDataSource, rightDataSource, }; } moveTo = (direction: TransferDirection) => { const { targetKeys = [], dataSource = [], onChange } = this.props; const { sourceSelectedKeys, targetSelectedKeys } = this.state; const moveKeys = direction === 'right' ? sourceSelectedKeys : targetSelectedKeys; // filter the disabled options const newMoveKeys = moveKeys.filter( (key: string) => !dataSource.some(data => !!(key === data.key && data.disabled)), ); // move items to target box const newTargetKeys = direction === 'right' ? newMoveKeys.concat(targetKeys) : targetKeys.filter(targetKey => newMoveKeys.indexOf(targetKey) === -1); // empty checked keys const oppositeDirection = direction === 'right' ? 'left' : 'right'; this.setState({ [this.getSelectedKeysName(oppositeDirection)]: [], }); this.handleSelectChange(oppositeDirection, []); if (onChange) { onChange(newTargetKeys, direction, newMoveKeys); } }; moveToLeft = () => this.moveTo('left'); moveToRight = () => this.moveTo('right'); handleSelectChange(direction: TransferDirection, holder: string[]) { const { sourceSelectedKeys, targetSelectedKeys } = this.state; const onSelectChange = this.props.onSelectChange; if (!onSelectChange) { return; } if (direction === 'left') { onSelectChange(holder, targetSelectedKeys); } else { onSelectChange(sourceSelectedKeys, holder); } } onItemSelectAll = (direction: TransferDirection, selectedKeys: string[], checkAll: boolean) => { const originalSelectedKeys = this.state[this.getSelectedKeysName(direction)] || []; let mergedCheckedKeys = []; if (checkAll) { // Merge current keys with origin key mergedCheckedKeys = Array.from(new Set([...originalSelectedKeys, ...selectedKeys])); } else { // Remove current keys from origin keys mergedCheckedKeys = originalSelectedKeys.filter( (key: string) => selectedKeys.indexOf(key) === -1, ); } this.handleSelectChange(direction, mergedCheckedKeys); if (!this.props.selectedKeys) { this.setState({ [this.getSelectedKeysName(direction)]: mergedCheckedKeys, }); } }; handleSelectAll = ( direction: TransferDirection, filteredDataSource: TransferItem[], checkAll: boolean, ) => { warning( false, 'Transfer', '`handleSelectAll` will be removed, please use `onSelectAll` instead.', ); this.onItemSelectAll(direction, filteredDataSource.map(({ key }) => key), !checkAll); }; // [Legacy] Old prop `body` pass origin check as arg. It's confusing. // TODO: Remove this in next version. handleLeftSelectAll = (filteredDataSource: TransferItem[], checkAll: boolean) => this.handleSelectAll('left', filteredDataSource, !checkAll); handleRightSelectAll = (filteredDataSource: TransferItem[], checkAll: boolean) => this.handleSelectAll('right', filteredDataSource, !checkAll); onLeftItemSelectAll = (selectedKeys: string[], checkAll: boolean) => this.onItemSelectAll('left', selectedKeys, checkAll); onRightItemSelectAll = (selectedKeys: string[], checkAll: boolean) => this.onItemSelectAll('right', selectedKeys, checkAll); handleFilter = (direction: TransferDirection, e: React.ChangeEvent) => { const { onSearchChange, onSearch } = this.props; const value = e.target.value; if (onSearchChange) { warning(false, 'Transfer', '`onSearchChange` is deprecated. Please use `onSearch` instead.'); onSearchChange(direction, e); } if (onSearch) { onSearch(direction, value); } }; handleLeftFilter = (e: React.ChangeEvent) => this.handleFilter('left', e); handleRightFilter = (e: React.ChangeEvent) => this.handleFilter('right', e); handleClear = (direction: TransferDirection) => { const { onSearch } = this.props; if (onSearch) { onSearch(direction, ''); } }; handleLeftClear = () => this.handleClear('left'); handleRightClear = () => this.handleClear('right'); onItemSelect = (direction: TransferDirection, selectedKey: string, checked: boolean) => { const { sourceSelectedKeys, targetSelectedKeys } = this.state; const holder = direction === 'left' ? [...sourceSelectedKeys] : [...targetSelectedKeys]; const index = holder.indexOf(selectedKey); if (index > -1) { holder.splice(index, 1); } if (checked) { holder.push(selectedKey); } this.handleSelectChange(direction, holder); if (!this.props.selectedKeys) { this.setState({ [this.getSelectedKeysName(direction)]: holder, }); } }; handleSelect = (direction: TransferDirection, selectedItem: TransferItem, checked: boolean) => { warning(false, 'Transfer', '`handleSelect` will be removed, please use `onSelect` instead.'); this.onItemSelect(direction, selectedItem.key, checked); }; handleLeftSelect = (selectedItem: TransferItem, checked: boolean) => this.handleSelect('left', selectedItem, checked); handleRightSelect = (selectedItem: TransferItem, checked: boolean) => this.handleSelect('right', selectedItem, checked); onLeftItemSelect = (selectedKey: string, checked: boolean) => this.onItemSelect('left', selectedKey, checked); onRightItemSelect = (selectedKey: string, checked: boolean) => this.onItemSelect('right', selectedKey, checked); handleScroll = (direction: TransferDirection, e: React.SyntheticEvent) => { const { onScroll } = this.props; if (onScroll) { onScroll(direction, e); } }; handleLeftScroll = (e: React.SyntheticEvent) => this.handleScroll('left', e); handleRightScroll = (e: React.SyntheticEvent) => this.handleScroll('right', e); getTitles(transferLocale: TransferLocale): string[] { const { props } = this; if (props.titles) { return props.titles; } return transferLocale.titles; } getSelectedKeysName(direction: TransferDirection) { return direction === 'left' ? 'sourceSelectedKeys' : 'targetSelectedKeys'; } getLocale = (transferLocale: TransferLocale, renderEmpty: RenderEmptyHandler) => { // Keep old locale props still working. const oldLocale: { notFoundContent?: any; searchPlaceholder?: string } = { notFoundContent: renderEmpty('Transfer'), }; if ('notFoundContent' in this.props) { oldLocale.notFoundContent = this.props.notFoundContent; } if ('searchPlaceholder' in this.props) { oldLocale.searchPlaceholder = this.props.searchPlaceholder; } return { ...transferLocale, ...oldLocale, ...this.props.locale }; }; renderTransfer = (transferLocale: TransferLocale) => ( {({ getPrefixCls, renderEmpty }: ConfigConsumerProps) => { const { prefixCls: customizePrefixCls, className, disabled, operations = [], showSearch, body, footer, style, listStyle, operationStyle, filterOption, render, lazy, children, showSelectAll, } = this.props; const prefixCls = getPrefixCls('transfer', customizePrefixCls); const locale = this.getLocale(transferLocale, renderEmpty); const { sourceSelectedKeys, targetSelectedKeys } = this.state; const { leftDataSource, rightDataSource } = this.separateDataSource(this.props); const leftActive = targetSelectedKeys.length > 0; const rightActive = sourceSelectedKeys.length > 0; const cls = classNames(className, prefixCls, { [`${prefixCls}-disabled`]: disabled, [`${prefixCls}-customize-list`]: !!children, }); const titles = this.getTitles(locale); return (
); }}
); render() { return ( {this.renderTransfer} ); } } polyfill(Transfer); export default Transfer;