ant-design/components/transfer/index.tsx

291 lines
8.5 KiB
TypeScript
Raw Normal View History

import * as React from 'react';
2016-08-08 14:12:02 +08:00
import { PropTypes } from 'react';
import List from './list';
2015-12-25 17:53:35 +08:00
import Operation from './operation';
import Search from './search';
2015-12-24 17:44:54 +08:00
import classNames from 'classnames';
2015-11-25 23:17:06 +08:00
function noop() {
}
export interface TransferItem {
2016-07-13 11:14:24 +08:00
key: number | string;
title: string;
description?: string;
chosen: boolean;
}
// Transfer
2016-07-07 20:25:03 +08:00
export interface TransferProps {
2016-07-13 11:14:24 +08:00
/** 数据源 */
dataSource: Array<TransferItem>;
/** 每行数据渲染函数 */
2016-07-13 17:22:23 +08:00
render?: (record: TransferItem) => any;
2016-07-13 11:14:24 +08:00
/** 显示在右侧框数据的key集合 */
targetKeys: Array<string>;
/** 变化时回调函数 */
2016-07-13 17:22:23 +08:00
onChange?: (targetKeys: Array<TransferItem>, direction: string, moveKeys: any) => void;
2016-07-13 11:14:24 +08:00
/** 两个穿梭框的自定义样式 */
listStyle?: React.CSSProperties;
/** 自定义类*/
2016-07-13 11:14:24 +08:00
className?: string;
/** 标题集合,顺序从左至右 */
titles?: Array<string>;
/** 操作文案集合,顺序从上至下 */
operations?: Array<string>;
/** 是否显示搜索框 */
showSearch?: boolean;
/** 搜索框的默认值 */
searchPlaceholder?: string;
/** 当列表为空时显示的内容 */
notFoundContent?: React.ReactNode | string;
/** 底部渲染函数 */
2016-07-13 17:22:23 +08:00
footer?: (props: any) => any;
2016-07-13 11:14:24 +08:00
style?: React.CSSProperties;
}
export default class Transfer extends React.Component<TransferProps, any> {
static List = List;
static Operation = Operation;
static Search = Search;
static defaultProps = {
prefixCls: 'ant-transfer',
dataSource: [],
render: noop,
targetKeys: [],
onChange: noop,
titles: ['源列表', '目的列表'],
operations: [],
showSearch: false,
body: noop,
footer: noop,
};
static propTypes = {
prefixCls: PropTypes.string,
dataSource: PropTypes.array,
render: PropTypes.func,
targetKeys: PropTypes.array,
onChange: PropTypes.func,
height: PropTypes.number,
listStyle: PropTypes.object,
className: PropTypes.string,
titles: PropTypes.array,
operations: PropTypes.array,
showSearch: PropTypes.bool,
filterOption: PropTypes.func,
searchPlaceholder: PropTypes.string,
notFoundContent: PropTypes.node,
body: PropTypes.func,
footer: PropTypes.func,
rowKey: PropTypes.func,
};
2015-11-25 23:17:06 +08:00
constructor(props) {
super(props);
this.state = {
2015-11-26 16:07:11 +08:00
leftFilter: '',
rightFilter: '',
2015-12-21 15:29:02 +08:00
leftCheckedKeys: [],
2015-12-23 19:41:56 +08:00
rightCheckedKeys: [],
2015-11-25 23:17:06 +08:00
};
}
2016-07-15 14:01:51 +08:00
componentWillReceiveProps(nextProps) {
const { leftCheckedKeys, rightCheckedKeys } = this.state;
if (nextProps.targetKeys !== this.props.targetKeys ||
nextProps.dataSource !== this.props.dataSource) {
// clear cached splited dataSource
this.splitedDataSource = null;
const { dataSource, targetKeys } = nextProps;
// clear key nolonger existed
// clear checkedKeys according to targetKeys
this.setState({
leftCheckedKeys: leftCheckedKeys
.filter(data => dataSource.filter(item => item.key === data).length)
.filter(data => targetKeys.filter(key => key === data).length === 0),
rightCheckedKeys: rightCheckedKeys
.filter(data => dataSource.filter(item => item.key === data).length)
.filter(data => targetKeys.filter(key => key === data).length > 0),
});
}
}
splitDataSource(props) {
if (this.splitedDataSource) {
return this.splitedDataSource;
}
const { targetKeys } = props;
let { dataSource } = props;
2015-11-26 11:18:37 +08:00
2016-06-02 16:15:19 +08:00
if (props.rowKey) {
dataSource = dataSource.map(record => {
record.key = props.rowKey(record);
return record;
});
}
2016-02-23 16:28:41 +08:00
let leftDataSource = [...dataSource];
2015-12-21 15:29:02 +08:00
let rightDataSource = [];
2015-12-17 16:08:16 +08:00
if (targetKeys.length > 0) {
2015-12-21 15:29:02 +08:00
targetKeys.forEach((targetKey) => {
rightDataSource.push(leftDataSource.filter((data, index) => {
if (data.key === targetKey) {
2015-12-21 15:29:02 +08:00
leftDataSource.splice(index, 1);
return true;
}
return false;
})[0]);
2015-12-17 16:08:16 +08:00
});
}
2015-12-21 15:29:02 +08:00
this.splitedDataSource = {
leftDataSource,
rightDataSource,
2015-12-21 15:29:02 +08:00
};
return this.splitedDataSource;
2015-12-16 23:02:49 +08:00
}
moveTo = (direction) => {
2015-12-21 15:29:02 +08:00
const { targetKeys } = this.props;
const { leftCheckedKeys, rightCheckedKeys } = this.state;
2016-02-01 14:07:17 +08:00
const moveKeys = direction === 'right' ? leftCheckedKeys : rightCheckedKeys;
2015-12-21 15:29:02 +08:00
// move items to target box
2016-02-01 14:07:17 +08:00
const newTargetKeys = direction === 'right'
? moveKeys.concat(targetKeys)
: targetKeys.filter(targetKey => !moveKeys.some(checkedKey => targetKey === checkedKey));
2015-12-21 15:29:02 +08:00
// empty checked keys
this.setState({
[direction === 'right' ? 'leftCheckedKeys' : 'rightCheckedKeys']: [],
2015-12-21 15:29:02 +08:00
});
2016-02-01 14:07:17 +08:00
this.props.onChange(newTargetKeys, direction, moveKeys);
2015-11-25 23:17:06 +08:00
}
2016-03-30 17:06:19 +08:00
moveToLeft = () => this.moveTo('left')
moveToRight = () => this.moveTo('right')
handleSelectAll = (direction, filteredDataSource, checkAll) => {
const holder = checkAll ? [] : filteredDataSource.map(item => item.key);
2015-11-25 23:17:06 +08:00
this.setState({
[`${direction}CheckedKeys`]: holder,
2015-11-25 23:17:06 +08:00
});
}
handleLeftSelectAll = (...args) => this.handleSelectAll('left', ...args)
handleRightSelectAll = (...args) => this.handleSelectAll('right', ...args)
2016-03-30 17:06:19 +08:00
handleFilter = (direction, e) => {
2015-12-21 15:29:02 +08:00
this.setState({
2015-12-23 19:41:56 +08:00
// add filter
[`${direction}Filter`]: e.target.value,
2015-12-21 15:29:02 +08:00
});
2015-11-26 16:07:11 +08:00
}
2016-03-30 17:06:19 +08:00
handleLeftFilter = (e) => this.handleFilter('left', e)
handleRightFilter = (e) => this.handleFilter('right', e)
handleClear = (direction) => {
2015-12-21 15:29:02 +08:00
this.setState({
[`${direction}Filter`]: '',
2015-11-25 23:17:06 +08:00
});
2015-12-21 15:29:02 +08:00
}
2015-11-25 23:17:06 +08:00
2016-03-30 17:06:19 +08:00
handleLeftClear = () => this.handleClear('left')
handleRightClear = () => this.handleClear('right')
handleSelect = (direction, selectedItem, checked) => {
2015-12-21 15:29:02 +08:00
const { leftCheckedKeys, rightCheckedKeys } = this.state;
const holder = direction === 'left' ? [...leftCheckedKeys] : [...rightCheckedKeys];
let index;
holder.forEach((key, i) => {
if (key === selectedItem.key) {
index = i;
}
});
if (index > -1) {
2015-12-21 15:29:02 +08:00
holder.splice(index, 1);
}
if (checked) {
2015-12-21 15:29:02 +08:00
holder.push(selectedItem.key);
}
2015-11-25 23:17:06 +08:00
this.setState({
[`${direction}CheckedKeys`]: holder,
2015-11-25 23:17:06 +08:00
});
}
2016-03-30 17:06:19 +08:00
handleLeftSelect = (selectedItem, checked) => this.handleSelect('left', selectedItem, checked);
handleRightSelect = (selectedItem, checked) => this.handleSelect('right', selectedItem, checked);
2015-11-25 23:17:06 +08:00
render() {
2015-12-24 17:44:54 +08:00
const {
prefixCls, titles, operations, showSearch, notFoundContent,
2015-12-24 17:44:54 +08:00
searchPlaceholder, body, footer, listStyle, className,
filterOption, render,
2015-12-24 17:44:54 +08:00
} = this.props;
2015-12-21 15:29:02 +08:00
const { leftFilter, rightFilter, leftCheckedKeys, rightCheckedKeys } = this.state;
2015-11-25 23:17:06 +08:00
const { leftDataSource, rightDataSource } = this.splitDataSource(this.props);
2015-12-23 19:41:56 +08:00
const leftActive = rightCheckedKeys.length > 0;
const rightActive = leftCheckedKeys.length > 0;
2015-12-24 17:44:54 +08:00
const cls = classNames({
[className]: !!className,
2016-03-29 17:19:31 +08:00
[prefixCls]: true,
2015-12-24 17:44:54 +08:00
});
return (
<div className={cls}>
<List titleText={titles[0]}
dataSource={leftDataSource}
filter={leftFilter}
filterOption={filterOption}
style={listStyle}
checkedKeys={leftCheckedKeys}
2016-03-30 17:06:19 +08:00
handleFilter={this.handleLeftFilter}
handleClear={this.handleLeftClear}
handleSelect={this.handleLeftSelect}
handleSelectAll={this.handleLeftSelectAll}
render={render}
showSearch={showSearch}
searchPlaceholder={searchPlaceholder}
notFoundContent={notFoundContent}
body={body}
footer={footer}
prefixCls={`${prefixCls}-list`}
/>
<Operation rightActive={rightActive}
rightArrowText={operations[0]}
2016-03-30 17:06:19 +08:00
moveToRight={this.moveToRight}
leftActive={leftActive}
leftArrowText={operations[1]}
2016-03-30 17:06:19 +08:00
moveToLeft={this.moveToLeft}
className={`${prefixCls}-operation`}
/>
<List titleText={titles[1]}
dataSource={rightDataSource}
filter={rightFilter}
filterOption={filterOption}
style={listStyle}
checkedKeys={rightCheckedKeys}
2016-03-30 17:06:19 +08:00
handleFilter={this.handleRightFilter}
handleClear={this.handleRightClear}
handleSelect={this.handleRightSelect}
handleSelectAll={this.handleRightSelectAll}
render={render}
showSearch={showSearch}
searchPlaceholder={searchPlaceholder}
notFoundContent={notFoundContent}
body={body}
footer={footer}
prefixCls={`${prefixCls}-list`}
/>
</div>
);
2015-11-25 23:17:06 +08:00
}
}