UNPKG

@alifd/next

Version:

A configurable component library for web built on React.

365 lines (317 loc) 15.9 kB
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); }