2016-03-26 17:22:39 +08:00
|
|
|
import React, { PropTypes } from 'react';
|
2016-05-12 11:09:27 +08:00
|
|
|
import List, { isRenderResultPlainObject } 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() {
|
|
|
|
}
|
|
|
|
|
2016-03-26 17:22:39 +08:00
|
|
|
export default class Transfer extends React.Component {
|
2016-03-29 14:01:10 +08:00
|
|
|
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,
|
2016-05-12 11:09:27 +08:00
|
|
|
};
|
2016-03-29 14:01:10 +08:00
|
|
|
|
|
|
|
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,
|
|
|
|
searchPlaceholder: PropTypes.string,
|
|
|
|
notFoundContent: PropTypes.node,
|
|
|
|
body: PropTypes.func,
|
|
|
|
footer: PropTypes.func,
|
2016-05-30 16:59:21 +08:00
|
|
|
rowKey: PropTypes.func,
|
2016-05-12 11:09:27 +08:00
|
|
|
};
|
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-05-06 12:09:26 +08:00
|
|
|
componentWillReceiveProps(nextProps) {
|
|
|
|
const { leftCheckedKeys, rightCheckedKeys } = this.state;
|
2016-06-21 17:41:39 +08:00
|
|
|
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),
|
|
|
|
});
|
|
|
|
}
|
2016-05-06 12:09:26 +08:00
|
|
|
}
|
|
|
|
splitDataSource(props) {
|
2016-06-21 17:41:39 +08:00
|
|
|
if (this.splitedDataSource) {
|
|
|
|
return this.splitedDataSource;
|
|
|
|
}
|
2016-05-30 16:59:21 +08:00
|
|
|
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) {
|
2016-05-30 16:59:21 +08:00
|
|
|
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
|
|
|
|
2016-01-07 17:46:46 +08:00
|
|
|
if (targetKeys.length > 0) {
|
2015-12-21 15:29:02 +08:00
|
|
|
targetKeys.forEach((targetKey) => {
|
2016-01-27 16:46:42 +08:00
|
|
|
rightDataSource.push(leftDataSource.filter((data, index) => {
|
2016-01-07 17:46:46 +08:00
|
|
|
if (data.key === targetKey) {
|
2015-12-21 15:29:02 +08:00
|
|
|
leftDataSource.splice(index, 1);
|
|
|
|
return true;
|
|
|
|
}
|
2016-02-22 10:52:30 +08:00
|
|
|
return false;
|
2016-01-27 16:46:42 +08:00
|
|
|
})[0]);
|
2015-12-17 16:08:16 +08:00
|
|
|
});
|
|
|
|
}
|
2015-12-21 15:29:02 +08:00
|
|
|
|
2016-06-21 17:41:39 +08:00
|
|
|
this.splitedDataSource = {
|
2016-01-07 17:46:46 +08:00
|
|
|
leftDataSource,
|
|
|
|
rightDataSource,
|
2015-12-21 15:29:02 +08:00
|
|
|
};
|
2016-06-21 17:41:39 +08:00
|
|
|
|
|
|
|
return this.splitedDataSource;
|
2015-12-16 23:02:49 +08:00
|
|
|
}
|
|
|
|
|
2016-03-23 19:50:44 +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({
|
2016-01-07 17:46:46 +08:00
|
|
|
[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')
|
|
|
|
|
2015-12-23 19:41:56 +08:00
|
|
|
getGlobalCheckStatus(direction) {
|
2016-05-06 12:09:26 +08:00
|
|
|
const { leftDataSource, rightDataSource } = this.splitDataSource(this.props);
|
2015-12-23 19:41:56 +08:00
|
|
|
const { leftFilter, rightFilter, leftCheckedKeys, rightCheckedKeys } = this.state;
|
|
|
|
|
2015-12-21 15:29:02 +08:00
|
|
|
const dataSource = direction === 'left' ? leftDataSource : rightDataSource;
|
2015-12-23 19:41:56 +08:00
|
|
|
const filter = direction === 'left' ? leftFilter : rightFilter;
|
|
|
|
const checkedKeys = direction === 'left' ? leftCheckedKeys : rightCheckedKeys;
|
|
|
|
const filteredDataSource = this.filterDataSource(dataSource, filter);
|
|
|
|
|
|
|
|
let globalCheckStatus;
|
|
|
|
|
2016-01-07 17:46:46 +08:00
|
|
|
if (checkedKeys.length > 0) {
|
|
|
|
if (checkedKeys.length < filteredDataSource.length) {
|
2015-12-23 19:41:56 +08:00
|
|
|
globalCheckStatus = 'part';
|
|
|
|
} else {
|
|
|
|
globalCheckStatus = 'all';
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
globalCheckStatus = 'none';
|
|
|
|
}
|
|
|
|
return globalCheckStatus;
|
|
|
|
}
|
|
|
|
|
|
|
|
filterDataSource(dataSource, filter) {
|
|
|
|
return dataSource.filter(item => {
|
2016-05-12 11:09:27 +08:00
|
|
|
const renderResult = this.props.render(item);
|
|
|
|
let itemText;
|
|
|
|
if (isRenderResultPlainObject(renderResult)) {
|
|
|
|
itemText = renderResult.value;
|
|
|
|
} else {
|
|
|
|
itemText = renderResult;
|
|
|
|
}
|
|
|
|
|
2015-12-24 17:44:54 +08:00
|
|
|
return this.matchFilter(itemText, filter);
|
2015-12-23 19:41:56 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
matchFilter(text, filterText) {
|
|
|
|
const regex = new RegExp(filterText);
|
|
|
|
return text.match(regex);
|
|
|
|
}
|
|
|
|
|
2016-03-23 19:50:44 +08:00
|
|
|
handleSelectAll = (direction) => {
|
2016-05-06 12:09:26 +08:00
|
|
|
const { leftDataSource, rightDataSource } = this.splitDataSource(this.props);
|
2015-12-23 19:41:56 +08:00
|
|
|
const { leftFilter, rightFilter } = this.state;
|
|
|
|
const dataSource = direction === 'left' ? leftDataSource : rightDataSource;
|
|
|
|
const filter = direction === 'left' ? leftFilter : rightFilter;
|
|
|
|
const checkStatus = this.getGlobalCheckStatus(direction);
|
2016-02-06 15:54:39 +08:00
|
|
|
const holder = (checkStatus === 'all') ? [] :
|
|
|
|
this.filterDataSource(dataSource, filter).map(item => item.key);
|
2015-11-25 23:17:06 +08:00
|
|
|
|
|
|
|
this.setState({
|
2016-02-17 18:04:42 +08:00
|
|
|
[`${direction}CheckedKeys`]: holder,
|
2015-11-25 23:17:06 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2016-03-30 17:06:19 +08:00
|
|
|
handleLeftSelectAll = () => this.handleSelectAll('left')
|
|
|
|
handleRightSelectAll = () => this.handleSelectAll('right')
|
|
|
|
|
2016-03-23 19:50:44 +08:00
|
|
|
handleFilter = (direction, e) => {
|
2015-12-21 15:29:02 +08:00
|
|
|
this.setState({
|
2015-12-23 19:41:56 +08:00
|
|
|
// deselect all
|
2016-02-17 18:04:42 +08:00
|
|
|
[`${direction}CheckedKeys`]: [],
|
2015-12-23 19:41:56 +08:00
|
|
|
// add filter
|
2016-02-17 18:04:42 +08:00
|
|
|
[`${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)
|
|
|
|
|
2016-03-23 19:50:44 +08:00
|
|
|
handleClear = (direction) => {
|
2015-12-21 15:29:02 +08:00
|
|
|
this.setState({
|
2016-02-17 18:04:42 +08:00
|
|
|
[`${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')
|
|
|
|
|
2016-03-23 19:50:44 +08:00
|
|
|
handleSelect = (direction, selectedItem, checked) => {
|
2015-12-21 15:29:02 +08:00
|
|
|
const { leftCheckedKeys, rightCheckedKeys } = this.state;
|
2016-06-21 17:41:39 +08:00
|
|
|
const holder = direction === 'left' ? [...leftCheckedKeys] : [...rightCheckedKeys];
|
2016-01-27 16:46:42 +08:00
|
|
|
let index;
|
|
|
|
holder.forEach((key, i) => {
|
|
|
|
if (key === selectedItem.key) {
|
|
|
|
index = i;
|
|
|
|
}
|
|
|
|
});
|
2016-01-07 17:46:46 +08:00
|
|
|
if (index > -1) {
|
2015-12-21 15:29:02 +08:00
|
|
|
holder.splice(index, 1);
|
|
|
|
}
|
2016-01-07 17:46:46 +08:00
|
|
|
if (checked) {
|
2015-12-21 15:29:02 +08:00
|
|
|
holder.push(selectedItem.key);
|
|
|
|
}
|
2015-11-25 23:17:06 +08:00
|
|
|
this.setState({
|
2016-02-17 18:04:42 +08:00
|
|
|
[`${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 {
|
2016-03-07 12:01:49 +08:00
|
|
|
prefixCls, titles, operations, showSearch, notFoundContent,
|
2015-12-24 17:44:54 +08:00
|
|
|
searchPlaceholder, body, footer, listStyle, className,
|
2016-05-12 11:09:27 +08:00
|
|
|
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
|
|
|
|
2016-05-06 12:09:26 +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;
|
|
|
|
|
|
|
|
const leftCheckStatus = this.getGlobalCheckStatus('left');
|
|
|
|
const rightCheckStatus = this.getGlobalCheckStatus('right');
|
2015-11-25 23:17:06 +08:00
|
|
|
|
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
|
|
|
});
|
|
|
|
|
2016-01-07 14:21:29 +08:00
|
|
|
return (
|
|
|
|
<div className={cls}>
|
|
|
|
<List titleText={titles[0]}
|
2016-01-07 17:46:46 +08:00
|
|
|
dataSource={leftDataSource}
|
|
|
|
filter={leftFilter}
|
|
|
|
style={listStyle}
|
|
|
|
checkedKeys={leftCheckedKeys}
|
|
|
|
checkStatus={leftCheckStatus}
|
2016-03-30 17:06:19 +08:00
|
|
|
handleFilter={this.handleLeftFilter}
|
|
|
|
handleClear={this.handleLeftClear}
|
|
|
|
handleSelect={this.handleLeftSelect}
|
|
|
|
handleSelectAll={this.handleLeftSelectAll}
|
2016-01-07 17:46:46 +08:00
|
|
|
position="left"
|
2016-05-12 11:09:27 +08:00
|
|
|
render={render}
|
2016-01-07 17:46:46 +08:00
|
|
|
showSearch={showSearch}
|
|
|
|
searchPlaceholder={searchPlaceholder}
|
2016-03-07 12:01:49 +08:00
|
|
|
notFoundContent={notFoundContent}
|
2016-01-07 17:46:46 +08:00
|
|
|
body={body}
|
|
|
|
footer={footer}
|
2016-06-06 13:54:10 +08:00
|
|
|
prefixCls={`${prefixCls}-list`}
|
|
|
|
/>
|
2016-01-07 14:21:29 +08:00
|
|
|
<Operation rightActive={rightActive}
|
2016-01-07 17:46:46 +08:00
|
|
|
rightArrowText={operations[0]}
|
2016-03-30 17:06:19 +08:00
|
|
|
moveToRight={this.moveToRight}
|
2016-01-07 17:46:46 +08:00
|
|
|
leftActive={leftActive}
|
|
|
|
leftArrowText={operations[1]}
|
2016-03-30 17:06:19 +08:00
|
|
|
moveToLeft={this.moveToLeft}
|
2016-06-06 13:54:10 +08:00
|
|
|
className={`${prefixCls}-operation`}
|
|
|
|
/>
|
2016-01-07 14:21:29 +08:00
|
|
|
<List titleText={titles[1]}
|
2016-01-07 17:46:46 +08:00
|
|
|
dataSource={rightDataSource}
|
|
|
|
filter={rightFilter}
|
|
|
|
style={listStyle}
|
|
|
|
checkedKeys={rightCheckedKeys}
|
|
|
|
checkStatus={rightCheckStatus}
|
2016-03-30 17:06:19 +08:00
|
|
|
handleFilter={this.handleRightFilter}
|
|
|
|
handleClear={this.handleRightClear}
|
|
|
|
handleSelect={this.handleRightSelect}
|
|
|
|
handleSelectAll={this.handleRightSelectAll}
|
2016-01-07 17:46:46 +08:00
|
|
|
position="right"
|
2016-05-12 11:09:27 +08:00
|
|
|
render={render}
|
2016-01-07 17:46:46 +08:00
|
|
|
showSearch={showSearch}
|
|
|
|
searchPlaceholder={searchPlaceholder}
|
2016-03-07 12:01:49 +08:00
|
|
|
notFoundContent={notFoundContent}
|
2016-01-07 17:46:46 +08:00
|
|
|
body={body}
|
|
|
|
footer={footer}
|
2016-06-06 13:54:10 +08:00
|
|
|
prefixCls={`${prefixCls}-list`}
|
|
|
|
/>
|
2016-01-07 14:21:29 +08:00
|
|
|
</div>
|
|
|
|
);
|
2015-11-25 23:17:06 +08:00
|
|
|
}
|
|
|
|
}
|