2018-03-25 10:07:04 +00:00
|
|
|
import PropTypes from '../../_util/vue-types'
|
|
|
|
import BaseMixin from '../../_util/BaseMixin'
|
|
|
|
import { connect } from '../../_util/store'
|
|
|
|
import TableRow from './TableRow'
|
|
|
|
import { remove } from './utils'
|
|
|
|
import { initDefaultProps, getOptionProps } from '../../_util/props-util'
|
|
|
|
|
|
|
|
export const ExpandableTableProps = () => ({
|
|
|
|
expandIconAsCell: PropTypes.bool,
|
2018-03-26 15:05:29 +00:00
|
|
|
expandRowByClick: PropTypes.bool,
|
2018-03-25 10:07:04 +00:00
|
|
|
expandedRowKeys: PropTypes.array,
|
|
|
|
expandedRowClassName: PropTypes.func,
|
|
|
|
defaultExpandAllRows: PropTypes.bool,
|
|
|
|
defaultExpandedRowKeys: PropTypes.array,
|
|
|
|
expandIconColumnIndex: PropTypes.number,
|
|
|
|
expandedRowRender: PropTypes.func,
|
|
|
|
childrenColumnName: PropTypes.string,
|
|
|
|
indentSize: PropTypes.number,
|
|
|
|
// onExpand: PropTypes.func,
|
|
|
|
// onExpandedRowsChange: PropTypes.func,
|
|
|
|
columnManager: PropTypes.object.isRequired,
|
|
|
|
store: PropTypes.object.isRequired,
|
|
|
|
prefixCls: PropTypes.string.isRequired,
|
|
|
|
data: PropTypes.array,
|
|
|
|
getRowKey: PropTypes.func,
|
|
|
|
})
|
|
|
|
|
|
|
|
const ExpandableTable = {
|
|
|
|
name: 'ExpandableTable',
|
|
|
|
mixins: [BaseMixin],
|
|
|
|
props: initDefaultProps(ExpandableTableProps(), {
|
|
|
|
expandIconAsCell: false,
|
|
|
|
expandedRowClassName: () => '',
|
|
|
|
expandIconColumnIndex: 0,
|
|
|
|
defaultExpandAllRows: false,
|
|
|
|
defaultExpandedRowKeys: [],
|
|
|
|
childrenColumnName: 'children',
|
|
|
|
indentSize: 15,
|
|
|
|
}),
|
|
|
|
|
|
|
|
data () {
|
|
|
|
const {
|
|
|
|
data,
|
|
|
|
childrenColumnName,
|
|
|
|
defaultExpandAllRows,
|
|
|
|
expandedRowKeys,
|
|
|
|
defaultExpandedRowKeys,
|
|
|
|
getRowKey,
|
|
|
|
} = this
|
|
|
|
|
|
|
|
let finnalExpandedRowKeys = []
|
|
|
|
let rows = [...data]
|
|
|
|
|
|
|
|
if (defaultExpandAllRows) {
|
|
|
|
for (let i = 0; i < rows.length; i++) {
|
|
|
|
const row = rows[i]
|
|
|
|
finnalExpandedRowKeys.push(getRowKey(row, i))
|
|
|
|
rows = rows.concat(row[childrenColumnName] || [])
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
finnalExpandedRowKeys = expandedRowKeys || defaultExpandedRowKeys
|
|
|
|
}
|
|
|
|
|
|
|
|
// this.columnManager = props.columnManager
|
|
|
|
// this.store = props.store
|
|
|
|
|
|
|
|
this.store.setState({
|
|
|
|
expandedRowsHeight: {},
|
|
|
|
expandedRowKeys: finnalExpandedRowKeys,
|
|
|
|
})
|
|
|
|
return {}
|
|
|
|
},
|
|
|
|
watch: {
|
|
|
|
expandedRowKeys (val) {
|
2018-09-05 13:28:54 +00:00
|
|
|
this.$nextTick(() => {
|
|
|
|
this.store.setState({
|
|
|
|
expandedRowKeys: val,
|
|
|
|
})
|
2018-03-25 10:07:04 +00:00
|
|
|
})
|
|
|
|
},
|
|
|
|
},
|
|
|
|
methods: {
|
|
|
|
handleExpandChange (expanded, record, event, rowKey, destroy = false) {
|
|
|
|
if (event) {
|
|
|
|
event.preventDefault()
|
|
|
|
event.stopPropagation()
|
|
|
|
}
|
|
|
|
|
|
|
|
let { expandedRowKeys } = this.store.getState()
|
|
|
|
|
|
|
|
if (expanded) {
|
|
|
|
// row was expaned
|
|
|
|
expandedRowKeys = [...expandedRowKeys, rowKey]
|
|
|
|
} else {
|
|
|
|
// row was collapse
|
|
|
|
const expandedRowIndex = expandedRowKeys.indexOf(rowKey)
|
|
|
|
if (expandedRowIndex !== -1) {
|
|
|
|
expandedRowKeys = remove(expandedRowKeys, rowKey)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!this.expandedRowKeys) {
|
|
|
|
this.store.setState({ expandedRowKeys })
|
|
|
|
}
|
|
|
|
this.__emit('expandedRowsChange', expandedRowKeys)
|
|
|
|
if (!destroy) {
|
|
|
|
this.__emit('expand', expanded, record)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
renderExpandIndentCell (rows, fixed) {
|
|
|
|
const { prefixCls, expandIconAsCell } = this
|
|
|
|
if (!expandIconAsCell || fixed === 'right' || !rows.length) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
const iconColumn = {
|
|
|
|
key: 'rc-table-expand-icon-cell',
|
|
|
|
className: `${prefixCls}-expand-icon-th`,
|
|
|
|
title: '',
|
|
|
|
rowSpan: rows.length,
|
|
|
|
}
|
|
|
|
|
|
|
|
rows[0].unshift({ ...iconColumn, column: iconColumn })
|
|
|
|
},
|
|
|
|
|
2018-04-01 04:52:27 +00:00
|
|
|
renderExpandedRow (record, index, expandedRowRender, className, ancestorKeys, indent, fixed) {
|
2018-03-25 10:07:04 +00:00
|
|
|
const { prefixCls, expandIconAsCell, indentSize } = this
|
2018-09-05 13:28:54 +00:00
|
|
|
const parentKey = ancestorKeys[ancestorKeys.length - 1]
|
|
|
|
const rowKey = `${parentKey}-extra-row`
|
|
|
|
const components = {
|
|
|
|
body: {
|
|
|
|
row: 'tr',
|
|
|
|
cell: 'td',
|
|
|
|
},
|
|
|
|
}
|
2018-03-25 10:07:04 +00:00
|
|
|
let colCount
|
|
|
|
if (fixed === 'left') {
|
|
|
|
colCount = this.columnManager.leftLeafColumns().length
|
|
|
|
} else if (fixed === 'right') {
|
|
|
|
colCount = this.columnManager.rightLeafColumns().length
|
|
|
|
} else {
|
|
|
|
colCount = this.columnManager.leafColumns().length
|
|
|
|
}
|
|
|
|
const columns = [{
|
|
|
|
key: 'extra-row',
|
2018-09-05 13:28:54 +00:00
|
|
|
customRender: () => {
|
|
|
|
const { expandedRowKeys } = this.store.getState()
|
|
|
|
const expanded = !!~expandedRowKeys.indexOf(parentKey)
|
|
|
|
return {
|
|
|
|
attrs: {
|
|
|
|
colSpan: colCount,
|
|
|
|
},
|
|
|
|
children: fixed !== 'right' ? expandedRowRender(record, index, indent, expanded) : ' ',
|
|
|
|
}
|
|
|
|
},
|
2018-03-25 10:07:04 +00:00
|
|
|
}]
|
|
|
|
if (expandIconAsCell && fixed !== 'right') {
|
|
|
|
columns.unshift({
|
|
|
|
key: 'expand-icon-placeholder',
|
2018-04-02 14:13:11 +00:00
|
|
|
customRender: () => null,
|
2018-03-25 10:07:04 +00:00
|
|
|
})
|
|
|
|
}
|
2018-09-05 13:28:54 +00:00
|
|
|
|
2018-03-25 10:07:04 +00:00
|
|
|
return (
|
|
|
|
<TableRow
|
|
|
|
key={rowKey}
|
|
|
|
columns={columns}
|
2018-03-25 14:23:04 +00:00
|
|
|
class={className}
|
2018-03-25 10:07:04 +00:00
|
|
|
rowKey={rowKey}
|
|
|
|
ancestorKeys={ancestorKeys}
|
|
|
|
prefixCls={`${prefixCls}-expanded-row`}
|
|
|
|
indentSize={indentSize}
|
|
|
|
indent={indent}
|
|
|
|
fixed={fixed}
|
|
|
|
components={components}
|
|
|
|
expandedRow
|
2018-03-25 14:23:04 +00:00
|
|
|
hasExpandIcon={() => {}}
|
2018-03-25 10:07:04 +00:00
|
|
|
/>
|
|
|
|
)
|
|
|
|
},
|
|
|
|
|
|
|
|
renderRows (renderRows, rows, record, index, indent, fixed, parentKey, ancestorKeys) {
|
|
|
|
const { expandedRowClassName, expandedRowRender, childrenColumnName } = this
|
|
|
|
const childrenData = record[childrenColumnName]
|
|
|
|
const nextAncestorKeys = [...ancestorKeys, parentKey]
|
|
|
|
const nextIndent = indent + 1
|
|
|
|
|
|
|
|
if (expandedRowRender) {
|
|
|
|
rows.push(
|
|
|
|
this.renderExpandedRow(
|
|
|
|
record,
|
|
|
|
index,
|
|
|
|
expandedRowRender,
|
|
|
|
expandedRowClassName(record, index, indent),
|
|
|
|
nextAncestorKeys,
|
|
|
|
nextIndent,
|
|
|
|
fixed,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (childrenData) {
|
2018-09-05 13:28:54 +00:00
|
|
|
renderRows(childrenData, nextIndent, rows, nextAncestorKeys)
|
2018-03-25 10:07:04 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
render () {
|
|
|
|
const { data, childrenColumnName, $scopedSlots, $listeners } = this
|
|
|
|
const props = getOptionProps(this)
|
|
|
|
const needIndentSpaced = data.some(record => record[childrenColumnName])
|
|
|
|
|
|
|
|
return $scopedSlots.default && $scopedSlots.default({
|
|
|
|
props,
|
|
|
|
on: $listeners,
|
|
|
|
needIndentSpaced,
|
|
|
|
renderRows: this.renderRows,
|
|
|
|
handleExpandChange: this.handleExpandChange,
|
|
|
|
renderExpandIndentCell: this.renderExpandIndentCell,
|
|
|
|
})
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
export default connect()(ExpandableTable)
|