328 lines
10 KiB
Vue
328 lines
10 KiB
Vue
<style lang="scss" scoped>
|
||
.u-table-row {
|
||
display: flex;
|
||
flex-direction: row;
|
||
position: relative;
|
||
}
|
||
|
||
// 添加border样式支持
|
||
.u-table-border {
|
||
border-top: 1px solid #ebeef5;
|
||
border-left: 1px solid #ebeef5;
|
||
border-right: 1px solid #ebeef5;
|
||
.u-table-cell {
|
||
border-right: 1px solid #ebeef5;
|
||
}
|
||
|
||
.u-table-cell:last-child {
|
||
border-right: none;
|
||
}
|
||
}
|
||
|
||
.u-table-cell {
|
||
flex: 1;
|
||
display: flex;
|
||
flex-direction: row;
|
||
align-items: center;
|
||
padding: 10px 1px;
|
||
font-size: 14px;
|
||
white-space: nowrap;
|
||
overflow: hidden;
|
||
text-overflow: ellipsis;
|
||
line-height: 1.1;
|
||
border-bottom: 1px solid #ebeef5;
|
||
&.u-text-left {
|
||
justify-content: flex-start;
|
||
text-align: left;
|
||
}
|
||
&.u-text-center {
|
||
justify-content: center;
|
||
text-align: center;
|
||
}
|
||
&.u-text-right {
|
||
justify-content: flex-end;
|
||
text-align: right;
|
||
}
|
||
}
|
||
|
||
.u-table-row-zebra {
|
||
background-color: #fafafa;
|
||
}
|
||
|
||
.u-table-row-highlight {
|
||
background-color: #f5f7fa;
|
||
}
|
||
|
||
.u-table-empty {
|
||
text-align: center;
|
||
padding: 20px;
|
||
color: #999;
|
||
}
|
||
|
||
// 隐藏被合并的单元格
|
||
.u-table-cell-hidden {
|
||
opacity: 0;
|
||
}
|
||
|
||
// 合并单元格样式
|
||
.u-table-cell-merged {
|
||
z-index: 1;
|
||
}
|
||
</style>
|
||
<template>
|
||
<view class="u-table-row u-table-row-child"
|
||
:class="[highlightCurrentRow && currentRow === row ? 'u-table-row-highlight' : '',
|
||
rowClassName ? rowClassName(row, rowIndex) : '',
|
||
stripe && rowIndex % 2 === 1 ? 'u-table-row-zebra' : ''
|
||
]" :style="{height: rowHeight}" @click="handleRowClick(row)">
|
||
<view v-for="(col, colIndex) in columns" :key="col.key"
|
||
class="u-table-cell"
|
||
:class="[col.align ? 'u-text-' + col.align : '',
|
||
cellClassName ? cellClassName(row, col) : '',
|
||
getFixedClass(col),
|
||
getCellSpanClass(rowIndex, colIndex)
|
||
]"
|
||
:style="[cellStyleInner({row: row, column: col, rowIndex: rowIndex, columnIndex: colIndex, level: level}), getCellSpanStyle(rowIndex, colIndex)]">
|
||
<!-- 复选框列 -->
|
||
<view v-if="col.type === 'selection'">
|
||
<checkbox :checked="isSelected(row)"
|
||
@click.stop="$emit('toggleSelect', row)" />
|
||
</view>
|
||
<template v-else>
|
||
<!-- 在mainCol列显示展开图标 -->
|
||
<view v-if="col.key === computedMainCol && hasTree"
|
||
@click.stop="toggleExpand(row)" :style="{width: expandWidth}">
|
||
<view v-if="row.children && row.children.length > 0">
|
||
{{ isExpanded(row) ? '▼' : '▶' }}
|
||
</view>
|
||
</view>
|
||
<slot name="cellChild" :row="row" :column="col" :prow="parentRow"
|
||
:rowIndex="rowIndex" :columnIndex="colIndex" :level="level">
|
||
<view class="u-table-cell_content">
|
||
{{ row[col.key] }}
|
||
</view>
|
||
</slot>
|
||
</template>
|
||
</view>
|
||
</view>
|
||
<!-- 递归渲染更深层的子级 -->
|
||
<template v-if="isExpanded(row) && row[treeProps.children] && row[treeProps.children].length">
|
||
<template v-for="(rowChild, childIndex) in row[treeProps.children]" :key="rowChild[rowKey] || childIndex">
|
||
<table-row
|
||
:row="rowChild"
|
||
:rowIndex="childIndex"
|
||
:parent-row="row"
|
||
:columns="columns"
|
||
:tree-props="treeProps"
|
||
:row-key="rowKey"
|
||
:expanded-keys="expandedKeys"
|
||
:cell-style-inner="cellStyleInner"
|
||
:is-expanded="isExpanded"
|
||
:row-class-name="rowClassName"
|
||
:stripe="stripe"
|
||
:cell-class-name="cellClassName"
|
||
:get-fixed-class="getFixedClass"
|
||
:highlight-current-row="highlightCurrentRow"
|
||
:current-row="currentRow"
|
||
:handle-row-click="handleRowClick"
|
||
:toggle-expand="toggleExpand"
|
||
:level="level + 1"
|
||
:rowHeight="rowHeight"
|
||
:hasTree="hasTree"
|
||
:selectedRows="selectedRows"
|
||
:expandWidth="expandWidth"
|
||
:computed-main-col="computedMainCol"
|
||
:span-method="spanMethod"
|
||
@toggle-select="$emit('toggleSelect', $event)"
|
||
@row-click="$emit('rowClick', $event)"
|
||
@toggle-expand="$emit('toggleExpand', $event)"
|
||
>
|
||
<template v-slot:cellChild="scope">
|
||
<slot name="cellChild" :row="scope.row" :column="scope.column" :prow="scope.prow"
|
||
:rowIndex="scope.rowIndex" :columnIndex="scope.columnIndex" :level="level">
|
||
</slot>
|
||
</template>
|
||
</table-row>
|
||
</template>
|
||
</template>
|
||
</template>
|
||
|
||
<script>
|
||
export default {
|
||
name: 'tableRow',
|
||
props: {
|
||
row: {
|
||
type: Object,
|
||
required: true
|
||
},
|
||
rowIndex: {
|
||
type: Number,
|
||
required: true
|
||
},
|
||
parentRow: {
|
||
type: Object,
|
||
default: null
|
||
},
|
||
columns: {
|
||
type: Array,
|
||
required: true
|
||
},
|
||
treeProps: {
|
||
type: Object,
|
||
required: true
|
||
},
|
||
rowKey: {
|
||
type: String,
|
||
required: true
|
||
},
|
||
expandedKeys: {
|
||
type: Array,
|
||
required: true
|
||
},
|
||
cellStyleInner: {
|
||
type: Function,
|
||
required: true
|
||
},
|
||
isExpanded: {
|
||
type: Function,
|
||
required: true
|
||
},
|
||
rowClassName: {
|
||
type: Function,
|
||
default: null
|
||
},
|
||
stripe: {
|
||
type: Boolean,
|
||
default: false
|
||
},
|
||
cellClassName: {
|
||
type: Function,
|
||
default: null
|
||
},
|
||
getFixedClass: {
|
||
type: Function,
|
||
required: true
|
||
},
|
||
highlightCurrentRow: {
|
||
type: Boolean,
|
||
default: false
|
||
},
|
||
currentRow: {
|
||
type: Object,
|
||
default: null
|
||
},
|
||
handleRowClick: {
|
||
type: Function,
|
||
required: true
|
||
},
|
||
toggleExpand: {
|
||
type: Function,
|
||
required: true
|
||
},
|
||
level: {
|
||
type: Number,
|
||
required: true
|
||
},
|
||
// 添加computedMainCol属性
|
||
computedMainCol: {
|
||
type: String,
|
||
required: true
|
||
},
|
||
expandWidth: {
|
||
type: String,
|
||
required: true
|
||
},
|
||
hasTree: {
|
||
type: Boolean,
|
||
required: false
|
||
},
|
||
selectedRows: {
|
||
type: Array,
|
||
required: false
|
||
},
|
||
rowHeight: {
|
||
type: String,
|
||
required: true
|
||
},
|
||
// 添加spanMethod属性
|
||
spanMethod: {
|
||
type: Function,
|
||
default: null
|
||
}
|
||
},
|
||
emits: ['rowClick', 'toggleExpand', 'toggleSelect'],
|
||
methods: {
|
||
isSelected(row) {
|
||
return this.selectedRows.some(r => r[this.rowKey] === row[this.rowKey]);
|
||
},
|
||
// 获取单元格的合并信息
|
||
getCellSpan(rowIndex, columnIndex) {
|
||
if (typeof this.spanMethod !== 'function') {
|
||
return { rowspan: 1, colspan: 1 };
|
||
}
|
||
|
||
const row = this.row;
|
||
const column = this.columns[columnIndex];
|
||
|
||
const result = this.spanMethod({
|
||
row,
|
||
column,
|
||
rowIndex,
|
||
columnIndex
|
||
});
|
||
|
||
if (Array.isArray(result)) {
|
||
const [rowspan, colspan] = result;
|
||
return { rowspan: rowspan != null ? rowspan : 1, colspan: colspan != null ? colspan : 1 };
|
||
} else if (typeof result === 'object') {
|
||
return {
|
||
rowspan: rowspan != null ? rowspan : 1,
|
||
colspan: colspan != null ? colspan : 1
|
||
};
|
||
}
|
||
|
||
return { rowspan: 1, colspan: 1 };
|
||
},
|
||
// 获取单元格的样式类
|
||
getCellSpanClass(rowIndex, columnIndex) {
|
||
const span = this.getCellSpan(rowIndex, columnIndex);
|
||
|
||
// 如果rowspan为0或colspan为0,表示该单元格被合并,需要隐藏
|
||
if (span.rowspan === 0 || span.colspan === 0) {
|
||
return 'u-table-cell-hidden';
|
||
} else if (span.rowspan > 1 || span.colspan > 1) {
|
||
// 如果有合并,添加合并样式类
|
||
return 'u-table-cell-merged';
|
||
}
|
||
|
||
return '';
|
||
},
|
||
// 获取单元格的样式
|
||
getCellSpanStyle(rowIndex, columnIndex) {
|
||
const span = this.getCellSpan(rowIndex, columnIndex);
|
||
const style = {};
|
||
|
||
// 设置rowspan
|
||
if (span.rowspan > 1) {
|
||
// 正确计算合并后的高度
|
||
const currentHeight = parseInt(this.rowHeight);
|
||
if (!isNaN(currentHeight)) {
|
||
style.height = `${span.rowspan * currentHeight}px`;
|
||
}
|
||
}
|
||
|
||
// 设置colspan
|
||
if (span.colspan > 1) {
|
||
style.flex = span.colspan;
|
||
}
|
||
|
||
// 如果rowspan为0或colspan为0,表示该单元格被合并,需要隐藏
|
||
if (span.rowspan === 0 || span.colspan === 0) {
|
||
style.display = 'none';
|
||
}
|
||
|
||
return style;
|
||
}
|
||
}
|
||
}
|
||
</script> |