@alifd/next
Version:
A configurable component library for web built on React.
365 lines (317 loc) • 15.9 kB
JavaScript
import _objectWithoutProperties from 'babel-runtime/helpers/objectWithoutProperties';
import _extends from 'babel-runtime/helpers/extends';
import _classCallCheck from 'babel-runtime/helpers/classCallCheck';
import _possibleConstructorReturn from 'babel-runtime/helpers/possibleConstructorReturn';
import _inherits from 'babel-runtime/helpers/inherits';
import React, { Children } from 'react';
import PropTypes from 'prop-types';
import { polyfill } from 'react-lifecycles-compat';
import Checkbox from '../checkbox';
import Radio from '../radio';
import { func, log } from '../util';
import zhCN from '../locale/zh-cn';
import SelectionRow from './selection/row';
import Col from './column';
import { statics } from './util';
var makeChain = func.makeChain;
var unique = function unique(arr) {
var key = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'this';
var temp = {},
ret = [];
arr.forEach(function (item) {
var value = void 0;
if (key === 'this') {
value = item;
} else {
value = item[key];
}
if (!temp[value]) {
ret.push(item);
temp[value] = true;
}
});
return ret;
};
export default function selection(BaseComponent) {
var _class, _temp;
/** Table */
var SelectionTable = (_temp = _class = function (_React$Component) {
_inherits(SelectionTable, _React$Component);
function SelectionTable(props, context) {
_classCallCheck(this, SelectionTable);
var _this = _possibleConstructorReturn(this, _React$Component.call(this, props, context));
_this.addSelection = function (columns) {
var _this$props = _this.props,
prefix = _this$props.prefix,
rowSelection = _this$props.rowSelection,
size = _this$props.size;
var attrs = rowSelection.columnProps && rowSelection.columnProps() || {};
if (!columns.find(function (record) {
return record.key === 'selection';
})) {
columns.unshift(_extends({
key: 'selection',
title: _this.renderSelectionHeader.bind(_this),
cell: _this.renderSelectionBody.bind(_this),
width: size === 'small' ? 34 : 50,
className: prefix + 'table-selection ' + prefix + 'table-prerow',
__normalized: true
}, attrs));
}
};
_this.renderSelectionHeader = function () {
var onChange = _this.selectAllRow,
attrs = {},
_this$props2 = _this.props,
rowSelection = _this$props2.rowSelection,
primaryKey = _this$props2.primaryKey,
dataSource = _this$props2.dataSource,
entireDataSource = _this$props2.entireDataSource,
locale = _this$props2.locale,
selectedRowKeys = _this.state.selectedRowKeys,
mode = rowSelection.mode ? rowSelection.mode : 'multiple';
var checked = !!selectedRowKeys.length;
var indeterminate = false;
var source = entireDataSource || dataSource;
_this.flatDataSource(source).filter(function (record, index) {
if (!rowSelection.getProps) {
return true;
} else {
return !(rowSelection.getProps(record, index) || {}).disabled;
}
}).map(function (record) {
return record[primaryKey];
}).forEach(function (id) {
if (selectedRowKeys.indexOf(id) === -1) {
checked = false;
} else {
indeterminate = true;
}
});
attrs.onClick = makeChain(function (e) {
e.stopPropagation();
}, attrs.onClick);
var userAttrs = rowSelection.titleProps && rowSelection.titleProps() || {};
if (checked) {
indeterminate = false;
}
return [mode === 'multiple' ? React.createElement(Checkbox, _extends({
key: '_total',
indeterminate: indeterminate,
'aria-label': locale.selectAll,
checked: checked,
onChange: onChange
}, attrs, userAttrs)) : null, rowSelection.titleAddons && rowSelection.titleAddons()];
};
_this.renderSelectionBody = function (value, index, record) {
var _this$props3 = _this.props,
rowSelection = _this$props3.rowSelection,
primaryKey = _this$props3.primaryKey;
var selectedRowKeys = _this.state.selectedRowKeys;
var mode = rowSelection.mode ? rowSelection.mode : 'multiple';
var checked = selectedRowKeys.indexOf(record[primaryKey]) > -1;
var onChange = _this.selectOneRow.bind(_this, index, record);
var attrs = rowSelection.getProps ? rowSelection.getProps(record, index) || {} : {};
attrs.onClick = makeChain(function (e) {
e.stopPropagation();
}, attrs.onClick);
return mode === 'multiple' ? React.createElement(Checkbox, _extends({ checked: checked, onChange: onChange }, attrs)) : React.createElement(Radio, _extends({ checked: checked, onChange: onChange }, attrs));
};
_this.selectAllRow = function (checked, e) {
var ret = [].concat(_this.state.selectedRowKeys),
_this$props4 = _this.props,
rowSelection = _this$props4.rowSelection,
primaryKey = _this$props4.primaryKey,
dataSource = _this$props4.dataSource,
entireDataSource = _this$props4.entireDataSource,
selectedRowKeys = _this.state.selectedRowKeys,
getProps = rowSelection.getProps;
var attrs = {},
records = [];
var source = entireDataSource ? entireDataSource : dataSource;
_this.flatDataSource(source).forEach(function (record, index) {
var id = record[primaryKey];
if (getProps) {
attrs = getProps(record, index) || {};
}
// 反选和全选的时候不要丢弃禁用项的选中状态
if (checked && (!attrs.disabled || selectedRowKeys.indexOf(id) > -1)) {
ret.push(id);
records.push(record);
} else if (attrs.disabled && selectedRowKeys.indexOf(id) > -1) {
ret.push(id);
records.push(record);
} else {
var i = ret.indexOf(id);
i > -1 && ret.splice(i, 1);
}
});
records = unique(records, primaryKey);
if (typeof rowSelection.onSelectAll === 'function') {
rowSelection.onSelectAll(checked, records);
}
_this.triggerSelection(rowSelection, unique(ret), records);
e.stopPropagation();
};
_this.state = {
selectedRowKeys: props.rowSelection && 'selectedRowKeys' in props.rowSelection ? props.rowSelection.selectedRowKeys || [] : []
};
return _this;
}
SelectionTable.prototype.getChildContext = function getChildContext() {
return {
rowSelection: this.props.rowSelection,
selectedRowKeys: this.state.selectedRowKeys
};
};
SelectionTable.getDerivedStateFromProps = function getDerivedStateFromProps(nextProps) {
if (nextProps.rowSelection && 'selectedRowKeys' in nextProps.rowSelection) {
var selectedRowKeys = nextProps.rowSelection.selectedRowKeys || [];
return {
selectedRowKeys: selectedRowKeys
};
}
return null;
};
SelectionTable.prototype.normalizeChildren = function normalizeChildren(children) {
var _props = this.props,
prefix = _props.prefix,
rowSelection = _props.rowSelection,
size = _props.size;
if (rowSelection) {
children = Children.map(children, function (child, index) {
return React.cloneElement(child, {
key: index
});
});
var attrs = rowSelection.columnProps && rowSelection.columnProps() || {};
children.unshift(React.createElement(Col, _extends({
key: 'selection',
title: this.renderSelectionHeader.bind(this),
cell: this.renderSelectionBody.bind(this),
width: size === 'small' ? 34 : 50,
className: prefix + 'table-selection ' + prefix + 'table-prerow',
__normalized: true
}, attrs)));
return children;
}
return children;
};
SelectionTable.prototype.selectOneRow = function selectOneRow(index, record, checked, e) {
var selectedRowKeys = [].concat(this.state.selectedRowKeys),
i = void 0;
var _props2 = this.props,
primaryKey = _props2.primaryKey,
rowSelection = _props2.rowSelection,
dataSource = _props2.dataSource,
entireDataSource = _props2.entireDataSource,
mode = rowSelection.mode ? rowSelection.mode : 'multiple',
id = record[primaryKey];
if (id === null || id === undefined) {
log.warning('Can\'t get value from record using given ' + primaryKey + ' as primaryKey.');
}
if (mode === 'multiple') {
if (checked) {
selectedRowKeys.push(id);
} else {
i = selectedRowKeys.indexOf(id);
selectedRowKeys.splice(i, 1);
}
} else if (checked) {
selectedRowKeys = [id];
}
var totalDS = dataSource;
if (Array.isArray(entireDataSource) && entireDataSource.length > dataSource.length) {
totalDS = entireDataSource;
}
var records = unique(totalDS.filter(function (item) {
return selectedRowKeys.indexOf(item[primaryKey]) > -1;
}), primaryKey);
if (typeof rowSelection.onSelect === 'function') {
rowSelection.onSelect(checked, record, records);
}
this.triggerSelection(rowSelection, selectedRowKeys, records);
e.stopPropagation();
};
SelectionTable.prototype.triggerSelection = function triggerSelection(rowSelection, selectedRowKeys, records) {
if (!('selectedRowKeys' in rowSelection)) {
this.setState({
selectedRowKeys: selectedRowKeys
});
}
if (typeof rowSelection.onChange === 'function') {
rowSelection.onChange(selectedRowKeys, records);
}
};
SelectionTable.prototype.flatDataSource = function flatDataSource(dataSource) {
var ret = dataSource;
var listHeader = this.context.listHeader;
if (listHeader) {
ret = [];
var hasChildrenSelection = listHeader.hasChildrenSelection,
hasSelection = listHeader.hasSelection;
dataSource.forEach(function (item) {
var children = item.children;
// 如果需要渲染selection才将这条记录插入到dataSource
// 或者没有孩子节点
if (hasSelection) {
ret.push(item);
}
if (children && hasChildrenSelection) {
ret = ret.concat(children);
}
});
}
return ret;
};
SelectionTable.prototype.render = function render() {
/* eslint-disable prefer-const */
var _props3 = this.props,
rowSelection = _props3.rowSelection,
components = _props3.components,
children = _props3.children,
columns = _props3.columns,
others = _objectWithoutProperties(_props3, ['rowSelection', 'components', 'children', 'columns']);
var useColumns = columns && !children;
if (rowSelection) {
if (useColumns) {
this.addSelection(columns);
} else {
children = this.normalizeChildren(children || []);
}
components = _extends({}, components);
components.Row = components.Row || SelectionRow;
}
return React.createElement(BaseComponent, _extends({}, others, { columns: columns, components: components, children: children }));
};
return SelectionTable;
}(React.Component), _class.SelectionRow = SelectionRow, _class.propTypes = _extends({
/**
* 是否启用选择模式
* @property {Function} getProps `Function(record, index)=>Object` 获取selection的默认属性
* @property {Function} onChange `Function(selectedRowKeys:Array, records:Array)` 选择改变的时候触发的事件,**注意:** 其中records只会包含当前dataSource的数据,很可能会小于selectedRowKeys的长度。
* @property {Function} onSelect `Function(selected:Boolean, record:Object, records:Array)` 用户手动选择/取消选择某行的回调
* @property {Function} onSelectAll `Function(selected:Boolean, records:Array)` 用户手动选择/取消选择所有行的回调
* @property {Array} selectedRowKeys 设置了此属性,将rowSelection变为受控状态,接收值为该行数据的primaryKey的值
* @property {String} mode 选择selection的模式, 可选值为`single`, `multiple`,默认为`multiple`
* @property {Function} columnProps `Function()=>Object` 选择列 的props,例如锁列、对齐等,可使用`Table.Column` 的所有参数
* @property {Function} titleProps `Function()=>Object` 选择列 表头的props,仅在 `multiple` 模式下生效
*/
rowSelection: PropTypes.object,
primaryKey: PropTypes.oneOfType([PropTypes.symbol, PropTypes.string]),
dataSource: PropTypes.array,
entireDataSource: PropTypes.array
}, BaseComponent.propTypes), _class.defaultProps = _extends({}, BaseComponent.defaultProps, {
locale: zhCN.Table,
primaryKey: 'id',
prefix: 'next-'
}), _class.contextTypes = {
listHeader: PropTypes.any
}, _class.childContextTypes = {
rowSelection: PropTypes.object,
selectedRowKeys: PropTypes.array
}, _temp);
SelectionTable.displayName = 'SelectionTable';
statics(SelectionTable, BaseComponent);
return polyfill(SelectionTable);
}