320 lines
8.9 KiB
JavaScript
320 lines
8.9 KiB
JavaScript
|
import { cellStarts, cellForced, defaultRenderCell, treeCellPrefix } from './config';
|
|||
|
import { mergeOptions, parseWidth, parseMinWidth, compose } from './util';
|
|||
|
import ElCheckbox from 'element-ui/packages/checkbox';
|
|||
|
|
|||
|
let columnIdSeed = 1;
|
|||
|
|
|||
|
export default {
|
|||
|
name: 'ElTableColumn',
|
|||
|
|
|||
|
props: {
|
|||
|
type: {
|
|||
|
type: String,
|
|||
|
default: 'default'
|
|||
|
},
|
|||
|
label: String,
|
|||
|
className: String,
|
|||
|
labelClassName: String,
|
|||
|
property: String,
|
|||
|
prop: String,
|
|||
|
width: {},
|
|||
|
minWidth: {},
|
|||
|
renderHeader: Function,
|
|||
|
sortable: {
|
|||
|
type: [Boolean, String],
|
|||
|
default: false
|
|||
|
},
|
|||
|
sortMethod: Function,
|
|||
|
sortBy: [String, Function, Array],
|
|||
|
resizable: {
|
|||
|
type: Boolean,
|
|||
|
default: true
|
|||
|
},
|
|||
|
columnKey: String,
|
|||
|
align: String,
|
|||
|
headerAlign: String,
|
|||
|
showTooltipWhenOverflow: Boolean,
|
|||
|
showOverflowTooltip: Boolean,
|
|||
|
fixed: [Boolean, String],
|
|||
|
formatter: Function,
|
|||
|
selectable: Function,
|
|||
|
reserveSelection: Boolean,
|
|||
|
filterMethod: Function,
|
|||
|
filteredValue: Array,
|
|||
|
filters: Array,
|
|||
|
filterPlacement: String,
|
|||
|
filterMultiple: {
|
|||
|
type: Boolean,
|
|||
|
default: true
|
|||
|
},
|
|||
|
index: [Number, Function],
|
|||
|
sortOrders: {
|
|||
|
type: Array,
|
|||
|
default() {
|
|||
|
return ['ascending', 'descending', null];
|
|||
|
},
|
|||
|
validator(val) {
|
|||
|
return val.every(order => ['ascending', 'descending', null].indexOf(order) > -1);
|
|||
|
}
|
|||
|
}
|
|||
|
},
|
|||
|
|
|||
|
data() {
|
|||
|
return {
|
|||
|
isSubColumn: false,
|
|||
|
columns: []
|
|||
|
};
|
|||
|
},
|
|||
|
|
|||
|
computed: {
|
|||
|
owner() {
|
|||
|
let parent = this.$parent;
|
|||
|
while (parent && !parent.tableId) {
|
|||
|
parent = parent.$parent;
|
|||
|
}
|
|||
|
return parent;
|
|||
|
},
|
|||
|
|
|||
|
columnOrTableParent() {
|
|||
|
let parent = this.$parent;
|
|||
|
while (parent && !parent.tableId && !parent.columnId) {
|
|||
|
parent = parent.$parent;
|
|||
|
}
|
|||
|
return parent;
|
|||
|
},
|
|||
|
|
|||
|
realWidth() {
|
|||
|
return parseWidth(this.width);
|
|||
|
},
|
|||
|
|
|||
|
realMinWidth() {
|
|||
|
return parseMinWidth(this.minWidth);
|
|||
|
},
|
|||
|
|
|||
|
realAlign() {
|
|||
|
return this.align ? 'is-' + this.align : null;
|
|||
|
},
|
|||
|
|
|||
|
realHeaderAlign() {
|
|||
|
return this.headerAlign ? 'is-' + this.headerAlign : this.realAlign;
|
|||
|
}
|
|||
|
},
|
|||
|
|
|||
|
methods: {
|
|||
|
getPropsData(...props) {
|
|||
|
return props.reduce((prev, cur) => {
|
|||
|
if (Array.isArray(cur)) {
|
|||
|
cur.forEach((key) => {
|
|||
|
prev[key] = this[key];
|
|||
|
});
|
|||
|
}
|
|||
|
return prev;
|
|||
|
}, {});
|
|||
|
},
|
|||
|
|
|||
|
getColumnElIndex(children, child) {
|
|||
|
return [].indexOf.call(children, child);
|
|||
|
},
|
|||
|
|
|||
|
setColumnWidth(column) {
|
|||
|
if (this.realWidth) {
|
|||
|
column.width = this.realWidth;
|
|||
|
}
|
|||
|
if (this.realMinWidth) {
|
|||
|
column.minWidth = this.realMinWidth;
|
|||
|
}
|
|||
|
if (!column.minWidth) {
|
|||
|
column.minWidth = 80;
|
|||
|
}
|
|||
|
column.realWidth = column.width === undefined ? column.minWidth : column.width;
|
|||
|
return column;
|
|||
|
},
|
|||
|
|
|||
|
setColumnForcedProps(column) {
|
|||
|
// 对于特定类型的 column,某些属性不允许设置
|
|||
|
const type = column.type;
|
|||
|
const source = cellForced[type] || {};
|
|||
|
Object.keys(source).forEach(prop => {
|
|||
|
let value = source[prop];
|
|||
|
if (value !== undefined) {
|
|||
|
column[prop] = prop === 'className' ? `${column[prop]} ${value}` : value;
|
|||
|
}
|
|||
|
});
|
|||
|
return column;
|
|||
|
},
|
|||
|
|
|||
|
setColumnRenders(column) {
|
|||
|
// renderHeader 属性不推荐使用。
|
|||
|
if (this.renderHeader) {
|
|||
|
console.warn('[Element Warn][TableColumn]Comparing to render-header, scoped-slot header is easier to use. We recommend users to use scoped-slot header.');
|
|||
|
} else if (column.type !== 'selection') {
|
|||
|
column.renderHeader = (h, scope) => {
|
|||
|
const renderHeader = this.$scopedSlots.header;
|
|||
|
return renderHeader ? renderHeader(scope) : column.label;
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
let originRenderCell = column.renderCell;
|
|||
|
// TODO: 这里的实现调整
|
|||
|
if (column.type === 'expand') {
|
|||
|
// 对于展开行,renderCell 不允许配置的。在上一步中已经设置过,这里需要简单封装一下。
|
|||
|
column.renderCell = (h, data) => (<div class="cell">
|
|||
|
{ originRenderCell(h, data) }
|
|||
|
</div>);
|
|||
|
this.owner.renderExpanded = (h, data) => {
|
|||
|
return this.$scopedSlots.default
|
|||
|
? this.$scopedSlots.default(data)
|
|||
|
: this.$slots.default;
|
|||
|
};
|
|||
|
} else {
|
|||
|
originRenderCell = originRenderCell || defaultRenderCell;
|
|||
|
// 对 renderCell 进行包装
|
|||
|
column.renderCell = (h, data) => {
|
|||
|
let children = null;
|
|||
|
if (this.$scopedSlots.default) {
|
|||
|
children = this.$scopedSlots.default(data);
|
|||
|
} else {
|
|||
|
children = originRenderCell(h, data);
|
|||
|
}
|
|||
|
const prefix = treeCellPrefix(h, data);
|
|||
|
const props = {
|
|||
|
class: 'cell',
|
|||
|
style: {}
|
|||
|
};
|
|||
|
if (column.showOverflowTooltip) {
|
|||
|
props.class += ' el-tooltip';
|
|||
|
props.style = {width: (data.column.realWidth || data.column.width) - 1 + 'px'};
|
|||
|
}
|
|||
|
return (<div { ...props }>
|
|||
|
{ prefix }
|
|||
|
{ children }
|
|||
|
</div>);
|
|||
|
};
|
|||
|
}
|
|||
|
return column;
|
|||
|
},
|
|||
|
|
|||
|
registerNormalWatchers() {
|
|||
|
const props = ['label', 'property', 'filters', 'filterMultiple', 'sortable', 'index', 'formatter', 'className', 'labelClassName', 'showOverflowTooltip'];
|
|||
|
// 一些属性具有别名
|
|||
|
const aliases = {
|
|||
|
prop: 'property',
|
|||
|
realAlign: 'align',
|
|||
|
realHeaderAlign: 'headerAlign',
|
|||
|
realWidth: 'width'
|
|||
|
};
|
|||
|
const allAliases = props.reduce((prev, cur) => {
|
|||
|
prev[cur] = cur;
|
|||
|
return prev;
|
|||
|
}, aliases);
|
|||
|
|
|||
|
Object.keys(allAliases).forEach(key => {
|
|||
|
const columnKey = aliases[key];
|
|||
|
|
|||
|
this.$watch(key, (newVal) => {
|
|||
|
this.columnConfig[columnKey] = newVal;
|
|||
|
});
|
|||
|
});
|
|||
|
},
|
|||
|
|
|||
|
registerComplexWatchers() {
|
|||
|
const props = ['fixed'];
|
|||
|
const aliases = {
|
|||
|
realWidth: 'width',
|
|||
|
realMinWidth: 'minWidth'
|
|||
|
};
|
|||
|
const allAliases = props.reduce((prev, cur) => {
|
|||
|
prev[cur] = cur;
|
|||
|
return prev;
|
|||
|
}, aliases);
|
|||
|
|
|||
|
Object.keys(allAliases).forEach(key => {
|
|||
|
const columnKey = aliases[key];
|
|||
|
|
|||
|
this.$watch(key, (newVal) => {
|
|||
|
this.columnConfig[columnKey] = newVal;
|
|||
|
const updateColumns = columnKey === 'fixed';
|
|||
|
this.owner.store.scheduleLayout(updateColumns);
|
|||
|
});
|
|||
|
});
|
|||
|
}
|
|||
|
},
|
|||
|
|
|||
|
components: {
|
|||
|
ElCheckbox
|
|||
|
},
|
|||
|
|
|||
|
beforeCreate() {
|
|||
|
this.row = {};
|
|||
|
this.column = {};
|
|||
|
this.$index = 0;
|
|||
|
this.columnId = '';
|
|||
|
},
|
|||
|
|
|||
|
created() {
|
|||
|
const parent = this.columnOrTableParent;
|
|||
|
this.isSubColumn = this.owner !== parent;
|
|||
|
this.columnId = (parent.tableId || parent.columnId) + '_column_' + columnIdSeed++;
|
|||
|
|
|||
|
const type = this.type || 'default';
|
|||
|
const sortable = this.sortable === '' ? true : this.sortable;
|
|||
|
const defaults = {
|
|||
|
...cellStarts[type],
|
|||
|
id: this.columnId,
|
|||
|
type: type,
|
|||
|
property: this.prop || this.property,
|
|||
|
align: this.realAlign,
|
|||
|
headerAlign: this.realHeaderAlign,
|
|||
|
showOverflowTooltip: this.showOverflowTooltip || this.showTooltipWhenOverflow,
|
|||
|
// filter 相关属性
|
|||
|
filterable: this.filters || this.filterMethod,
|
|||
|
filteredValue: [],
|
|||
|
filterPlacement: '',
|
|||
|
isColumnGroup: false,
|
|||
|
filterOpened: false,
|
|||
|
// sort 相关属性
|
|||
|
sortable: sortable,
|
|||
|
// index 列
|
|||
|
index: this.index
|
|||
|
};
|
|||
|
|
|||
|
const basicProps = ['columnKey', 'label', 'className', 'labelClassName', 'type', 'renderHeader', 'formatter', 'fixed', 'resizable'];
|
|||
|
const sortProps = ['sortMethod', 'sortBy', 'sortOrders'];
|
|||
|
const selectProps = ['selectable', 'reserveSelection'];
|
|||
|
const filterProps = ['filterMethod', 'filters', 'filterMultiple', 'filterOpened', 'filteredValue', 'filterPlacement'];
|
|||
|
|
|||
|
let column = this.getPropsData(basicProps, sortProps, selectProps, filterProps);
|
|||
|
column = mergeOptions(defaults, column);
|
|||
|
|
|||
|
// 注意 compose 中函数执行的顺序是从右到左
|
|||
|
const chains = compose(this.setColumnRenders, this.setColumnWidth, this.setColumnForcedProps);
|
|||
|
column = chains(column);
|
|||
|
|
|||
|
this.columnConfig = column;
|
|||
|
|
|||
|
// 注册 watcher
|
|||
|
this.registerNormalWatchers();
|
|||
|
this.registerComplexWatchers();
|
|||
|
},
|
|||
|
|
|||
|
mounted() {
|
|||
|
const owner = this.owner;
|
|||
|
const parent = this.columnOrTableParent;
|
|||
|
const children = this.isSubColumn ? parent.$el.children : parent.$refs.hiddenColumns.children;
|
|||
|
const columnIndex = this.getColumnElIndex(children, this.$el);
|
|||
|
|
|||
|
owner.store.commit('insertColumn', this.columnConfig, columnIndex, this.isSubColumn ? parent.columnConfig : null);
|
|||
|
},
|
|||
|
|
|||
|
destroyed() {
|
|||
|
if (!this.$parent) return;
|
|||
|
const parent = this.$parent;
|
|||
|
this.owner.store.commit('removeColumn', this.columnConfig, this.isSubColumn ? parent.columnConfig : null);
|
|||
|
},
|
|||
|
|
|||
|
render(h) {
|
|||
|
// slots 也要渲染,需要计算合并表头
|
|||
|
return h('div', this.$slots.default);
|
|||
|
}
|
|||
|
};
|