2015-11-25 23:17:06 +08:00
|
|
|
import React, { Component, PropTypes } from 'react';
|
|
|
|
import List from './list.jsx';
|
2015-11-26 16:07:11 +08:00
|
|
|
import Operation from './operation.jsx';
|
2015-12-23 19:41:56 +08:00
|
|
|
import Search from './search.jsx';
|
2015-11-25 23:17:06 +08:00
|
|
|
|
|
|
|
function noop() {
|
|
|
|
}
|
|
|
|
|
|
|
|
class Transfer extends Component {
|
|
|
|
|
|
|
|
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
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2015-12-21 15:29:02 +08:00
|
|
|
splitDataSource() {
|
|
|
|
const { targetKeys, dataSource } = this.props;
|
2015-11-26 11:18:37 +08:00
|
|
|
|
2015-12-21 15:29:02 +08:00
|
|
|
let leftDataSource = Object.assign([], dataSource);
|
|
|
|
let rightDataSource = [];
|
2015-12-17 16:08:16 +08:00
|
|
|
|
2015-12-21 15:29:02 +08:00
|
|
|
if ( targetKeys.length > 0 ) {
|
|
|
|
targetKeys.forEach((targetKey) => {
|
|
|
|
rightDataSource.push(leftDataSource.find((data, index) => {
|
|
|
|
if( data.key === targetKey ) {
|
|
|
|
leftDataSource.splice(index, 1);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}));
|
2015-12-17 16:08:16 +08:00
|
|
|
});
|
|
|
|
}
|
2015-12-21 15:29:02 +08:00
|
|
|
|
|
|
|
return {
|
|
|
|
leftDataSource: leftDataSource,
|
|
|
|
rightDataSource: rightDataSource,
|
|
|
|
};
|
2015-12-16 23:02:49 +08:00
|
|
|
}
|
|
|
|
|
2015-11-25 23:17:06 +08:00
|
|
|
moveTo(direction) {
|
2015-12-21 15:29:02 +08:00
|
|
|
const { targetKeys } = this.props;
|
|
|
|
const { leftCheckedKeys, rightCheckedKeys } = this.state;
|
|
|
|
// move items to target box
|
|
|
|
const newTargetKeys = direction === 'right' ?
|
|
|
|
leftCheckedKeys.concat(targetKeys) :
|
|
|
|
targetKeys.filter((targetKey) => !rightCheckedKeys.some((checkedKey) => targetKey === checkedKey));
|
|
|
|
|
|
|
|
// empty checked keys
|
|
|
|
this.setState({
|
|
|
|
[direction === 'right' ? 'leftCheckedKeys' : 'rightCheckedKeys']: [],
|
|
|
|
});
|
|
|
|
|
|
|
|
this.props.onChange(newTargetKeys);
|
2015-11-25 23:17:06 +08:00
|
|
|
}
|
|
|
|
|
2015-12-23 19:41:56 +08:00
|
|
|
getGlobalCheckStatus(direction) {
|
2015-12-21 15:29:02 +08:00
|
|
|
const { leftDataSource, rightDataSource } = this.splitDataSource();
|
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;
|
|
|
|
|
|
|
|
if ( checkedKeys.length > 0 ) {
|
|
|
|
if ( checkedKeys.length < filteredDataSource.length ) {
|
|
|
|
globalCheckStatus = 'part';
|
|
|
|
} else {
|
|
|
|
globalCheckStatus = 'all';
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
globalCheckStatus = 'none';
|
|
|
|
}
|
|
|
|
return globalCheckStatus;
|
|
|
|
}
|
|
|
|
|
|
|
|
filterDataSource(dataSource, filter) {
|
|
|
|
const self = this;
|
|
|
|
return dataSource.filter(item => {
|
|
|
|
const itemText = self.props.render(item);
|
|
|
|
return self.matchFilter(itemText, filter);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
matchFilter(text, filterText) {
|
|
|
|
const regex = new RegExp(filterText);
|
|
|
|
return text.match(regex);
|
|
|
|
}
|
|
|
|
|
|
|
|
handleSelectAll(direction) {
|
|
|
|
const { leftDataSource, rightDataSource } = this.splitDataSource();
|
|
|
|
const { leftFilter, rightFilter } = this.state;
|
|
|
|
const dataSource = direction === 'left' ? leftDataSource : rightDataSource;
|
|
|
|
const filter = direction === 'left' ? leftFilter : rightFilter;
|
|
|
|
const checkStatus = this.getGlobalCheckStatus(direction);
|
2015-12-21 15:29:02 +08:00
|
|
|
let holder = [];
|
|
|
|
|
2015-12-23 19:41:56 +08:00
|
|
|
if ( checkStatus === 'all' ) {
|
2015-12-21 15:29:02 +08:00
|
|
|
holder = [];
|
|
|
|
} else {
|
2015-12-23 19:41:56 +08:00
|
|
|
holder = this.filterDataSource(dataSource, filter).map(item => item.key);
|
2015-11-25 23:17:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
this.setState({
|
2015-12-21 15:29:02 +08:00
|
|
|
[direction === 'left' ? 'leftCheckedKeys' : 'rightCheckedKeys']: holder,
|
2015-11-25 23:17:06 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2015-11-26 16:07:11 +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
|
|
|
|
[direction === 'left' ? 'leftCheckedKeys' : 'rightCheckedKeys']: [],
|
|
|
|
// add filter
|
2015-12-21 15:29:02 +08:00
|
|
|
[direction === 'left' ? 'leftFilter' : 'rightFilter']: e.target.value,
|
|
|
|
});
|
2015-11-26 16:07:11 +08:00
|
|
|
}
|
|
|
|
|
2015-12-17 16:08:16 +08:00
|
|
|
handleClear(direction) {
|
2015-12-21 15:29:02 +08:00
|
|
|
this.setState({
|
|
|
|
[direction === 'left' ? 'leftFilter' : 'rightFilter']: '',
|
2015-11-25 23:17:06 +08:00
|
|
|
});
|
2015-12-21 15:29:02 +08:00
|
|
|
}
|
2015-11-25 23:17:06 +08:00
|
|
|
|
2015-12-21 15:29:02 +08:00
|
|
|
handleSelect(direction, selectedItem, checked) {
|
|
|
|
const { leftCheckedKeys, rightCheckedKeys } = this.state;
|
|
|
|
const holder = direction === 'left' ? leftCheckedKeys : rightCheckedKeys;
|
|
|
|
const index = holder.findIndex((key) => key === selectedItem.key);
|
|
|
|
if ( index > -1 ) {
|
|
|
|
holder.splice(index, 1);
|
|
|
|
}
|
|
|
|
if ( checked ) {
|
|
|
|
holder.push(selectedItem.key);
|
|
|
|
}
|
2015-11-25 23:17:06 +08:00
|
|
|
this.setState({
|
2015-12-21 15:29:02 +08:00
|
|
|
[direction === 'left' ? 'leftCheckedKeys' : 'rightCheckedKeys']: holder,
|
2015-11-25 23:17:06 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
2015-12-23 20:05:55 +08:00
|
|
|
const { prefixCls, titles, operations, showSearch, searchPlaceholder, body, footer, boxWidth } = 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
|
|
|
|
2015-12-21 15:29:02 +08:00
|
|
|
const { leftDataSource, rightDataSource } = this.splitDataSource();
|
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
|
|
|
|
|
|
|
return <div className={prefixCls}>
|
2015-12-21 15:29:02 +08:00
|
|
|
<List title={titles[0]}
|
2015-12-16 23:02:49 +08:00
|
|
|
dataSource={leftDataSource}
|
2015-11-26 16:07:11 +08:00
|
|
|
filter={leftFilter}
|
2015-12-23 20:05:55 +08:00
|
|
|
width={boxWidth}
|
2015-12-21 15:29:02 +08:00
|
|
|
checkedKeys={leftCheckedKeys}
|
2015-12-23 19:41:56 +08:00
|
|
|
checkStatus={leftCheckStatus}
|
2015-11-26 16:07:11 +08:00
|
|
|
handleFilter={this.handleFilter.bind(this, 'left')}
|
2015-12-17 16:08:16 +08:00
|
|
|
handleClear={this.handleClear.bind(this, 'left')}
|
2015-12-21 15:29:02 +08:00
|
|
|
handleSelect={this.handleSelect.bind(this, 'left')}
|
2015-11-26 16:07:11 +08:00
|
|
|
handleSelectAll={this.handleSelectAll.bind(this, 'left')}
|
2015-12-16 23:02:49 +08:00
|
|
|
position="left"
|
2015-12-21 15:29:02 +08:00
|
|
|
render={this.props.render}
|
2015-12-15 17:08:34 +08:00
|
|
|
showSearch={showSearch}
|
2015-12-21 15:29:02 +08:00
|
|
|
searchPlaceholder={searchPlaceholder}
|
2015-12-15 17:08:34 +08:00
|
|
|
body={body}
|
|
|
|
footer={footer}
|
|
|
|
/>
|
2015-12-21 15:29:02 +08:00
|
|
|
<Operation rightActive={rightActive} rightArrowText={operations[0]} moveToRight={this.moveTo.bind(this, 'right')}
|
|
|
|
leftActive={leftActive} leftArrowText={operations[1]} moveToLeft={this.moveTo.bind(this, 'left')} />
|
|
|
|
<List title={titles[1]}
|
2015-12-16 23:02:49 +08:00
|
|
|
dataSource={rightDataSource}
|
2015-11-26 16:07:11 +08:00
|
|
|
filter={rightFilter}
|
2015-12-23 20:05:55 +08:00
|
|
|
width={boxWidth}
|
2015-12-21 15:29:02 +08:00
|
|
|
checkedKeys={rightCheckedKeys}
|
2015-12-23 19:41:56 +08:00
|
|
|
checkStatus={rightCheckStatus}
|
2015-11-26 16:07:11 +08:00
|
|
|
handleFilter={this.handleFilter.bind(this, 'right')}
|
2015-12-17 16:08:16 +08:00
|
|
|
handleClear={this.handleClear.bind(this, 'right')}
|
2015-12-21 15:29:02 +08:00
|
|
|
handleSelect={this.handleSelect.bind(this, 'right')}
|
2015-11-26 16:07:11 +08:00
|
|
|
handleSelectAll={this.handleSelectAll.bind(this, 'right')}
|
2015-12-16 23:02:49 +08:00
|
|
|
position="right"
|
2015-12-21 15:29:02 +08:00
|
|
|
render={this.props.render}
|
2015-12-15 17:08:34 +08:00
|
|
|
showSearch={showSearch}
|
2015-12-21 15:29:02 +08:00
|
|
|
searchPlaceholder={searchPlaceholder}
|
2015-12-15 17:08:34 +08:00
|
|
|
body={body}
|
|
|
|
footer={footer}
|
|
|
|
/>
|
2015-11-25 23:17:06 +08:00
|
|
|
</div>;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Transfer.defaultProps = {
|
|
|
|
prefixCls: 'ant-transfer',
|
2015-11-26 11:18:37 +08:00
|
|
|
dataSource: [],
|
2015-12-21 15:29:02 +08:00
|
|
|
render: noop,
|
|
|
|
targetKeys: [],
|
2015-11-25 23:17:06 +08:00
|
|
|
onChange: noop,
|
2015-12-23 20:05:55 +08:00
|
|
|
boxWidth: 160,
|
2015-12-21 15:29:02 +08:00
|
|
|
titles: ['源列表', '目的列表'],
|
|
|
|
operations: [],
|
2015-11-25 23:17:06 +08:00
|
|
|
showSearch: false,
|
|
|
|
searchPlaceholder: '请输入搜索内容',
|
2015-12-15 17:08:34 +08:00
|
|
|
body: noop,
|
2015-12-21 15:29:02 +08:00
|
|
|
footer: noop,
|
2015-11-25 23:17:06 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
Transfer.propTypes = {
|
|
|
|
prefixCls: PropTypes.string,
|
|
|
|
dataSource: PropTypes.array,
|
2015-12-21 15:29:02 +08:00
|
|
|
render: PropTypes.func,
|
|
|
|
targetKeys: PropTypes.array,
|
|
|
|
onChange: PropTypes.func,
|
2015-12-23 20:05:55 +08:00
|
|
|
boxWidth: PropTypes.number,
|
2015-12-21 15:29:02 +08:00
|
|
|
titles: PropTypes.array,
|
|
|
|
operations: PropTypes.array,
|
2015-11-25 23:17:06 +08:00
|
|
|
showSearch: PropTypes.bool,
|
|
|
|
searchPlaceholder: PropTypes.string,
|
2015-12-21 15:29:02 +08:00
|
|
|
body: PropTypes.func,
|
|
|
|
footer: PropTypes.func,
|
2015-11-25 23:17:06 +08:00
|
|
|
};
|
|
|
|
|
2015-12-23 19:41:56 +08:00
|
|
|
Transfer.List = List;
|
|
|
|
Transfer.Operation = Operation;
|
|
|
|
Transfer.Search = Search;
|
|
|
|
|
2015-11-25 23:17:06 +08:00
|
|
|
export default Transfer;
|