756 lines
26 KiB
Vue
756 lines
26 KiB
Vue
|
|
<template>
|
|||
|
|
<view class="u-table2" :class="{ 'u-table-border': border }">
|
|||
|
|
<scroll-view scroll-x scroll-y class="u-table2-content"
|
|||
|
|
:style="{ height: height ? addUnit(height) : 'auto' }"
|
|||
|
|
@scroll="onScroll">
|
|||
|
|
<!-- 表头 -->
|
|||
|
|
<view v-if="showHeader" class="u-table-header"
|
|||
|
|
:class="{ 'u-table-sticky': fixedHeader }"
|
|||
|
|
:style="{minWidth: scrollWidth}">
|
|||
|
|
<view class="u-table-row">
|
|||
|
|
<view v-for="(col, colIndex) in columns" :key="col.key" class="u-table-cell"
|
|||
|
|
:class="[col.headerAlign ? 'u-text-' + col.headerAlign : (col.align ? 'u-text-' + col.align : '') ,
|
|||
|
|
headerCellClassName ? headerCellClassName(col) : '',
|
|||
|
|
getFixedClass(col)
|
|||
|
|
]" :style="headerColStyle(col)" @click="handleHeaderClick(col)">
|
|||
|
|
<slot name="header" :column="col" :columnIndex="colIndex" :level="1">
|
|||
|
|
</slot>
|
|||
|
|
<text v-if="!$slots['header']">{{ col.title }}</text>
|
|||
|
|
<template v-if="col.sortable">
|
|||
|
|
<slot name="headerSort" :sortStatus="getSortValue(col.key)" :column="col"
|
|||
|
|
:columnIndex="colIndex" :level="1">
|
|||
|
|
</slot>
|
|||
|
|
<view v-if="!$slots['headerSort']">
|
|||
|
|
{{ getSortIcon(col.key) }}
|
|||
|
|
</view>
|
|||
|
|
</template>
|
|||
|
|
</view>
|
|||
|
|
</view>
|
|||
|
|
</view>
|
|||
|
|
|
|||
|
|
<!-- 表体 -->
|
|||
|
|
<view class="u-table-body" :style="{ minWidth: scrollWidth, maxHeight: maxHeight ? maxHeight + 'px' : 'none' }">
|
|||
|
|
<template v-if="data && data.length > 0">
|
|||
|
|
<table-row
|
|||
|
|
v-for="(row, rowIndex) in sortedData"
|
|||
|
|
:key="row[rowKey] || rowIndex"
|
|||
|
|
:row="row"
|
|||
|
|
:rowIndex="rowIndex"
|
|||
|
|
:parent-row="null"
|
|||
|
|
: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="1"
|
|||
|
|
:rowHeight="rowHeight"
|
|||
|
|
:hasTree="hasTree"
|
|||
|
|
:selectedRows="selectedRows"
|
|||
|
|
:expandWidth="expandWidth"
|
|||
|
|
:computedMainCol="computedMainCol"
|
|||
|
|
:span-method="spanMethod"
|
|||
|
|
@toggle-select="toggleSelect"
|
|||
|
|
@row-click="handleRowClick"
|
|||
|
|
@toggle-expand="toggleExpand"
|
|||
|
|
>
|
|||
|
|
<template v-slot:cellChild="scope">
|
|||
|
|
<slot name="cell" :row="scope.row" :column="scope.column" :prow="scope.prow"
|
|||
|
|
:rowIndex="scope.rowIndex" :columnIndex="scope.columnIndex" :level="scope.level">
|
|||
|
|
</slot>
|
|||
|
|
</template>
|
|||
|
|
</table-row>
|
|||
|
|
</template>
|
|||
|
|
<template v-else>
|
|||
|
|
<slot name="empty">
|
|||
|
|
</slot>
|
|||
|
|
<view v-if="!$slots['empty']" class="u-table-empty">{{ emptyText }}</view>
|
|||
|
|
</template>
|
|||
|
|
</view>
|
|||
|
|
</scroll-view>
|
|||
|
|
|
|||
|
|
<!-- 固定列浮动视图 -->
|
|||
|
|
<view v-if="showFixedColumnShadow" class="u-table-fixed-shadow" :style="{ height: tableHeight }">
|
|||
|
|
<!-- 表头 -->
|
|||
|
|
<view v-if="showHeader" class="u-table-header" :class="{ 'u-table-sticky': fixedHeader }" :style="{minWidth: scrollWidth}">
|
|||
|
|
<view class="u-table-row" :style="{height: headerHeight}">
|
|||
|
|
<view v-for="(col, colIndex) in visibleFixedLeftColumns" :key="col.key" class="u-table-cell"
|
|||
|
|
:style="headerColStyle(col)"
|
|||
|
|
:class="[col.align ? 'u-text-' + col.align : '',
|
|||
|
|
headerCellClassName ? headerCellClassName(col) : '',
|
|||
|
|
getFixedClass(col)
|
|||
|
|
]" @click="handleHeaderClick(col)">
|
|||
|
|
<slot name="header" :column="col" :columnIndex="colIndex" :level="1">
|
|||
|
|
</slot>
|
|||
|
|
<text v-if="!$slots['header']">{{ col.title }}</text>
|
|||
|
|
<template v-if="col.sortable">
|
|||
|
|
<slot name="headerSort" :sortStatus="getSortValue(col.key)" :column="col"
|
|||
|
|
:columnIndex="colIndex" :level="1">
|
|||
|
|
</slot>
|
|||
|
|
<view v-if="!$slots['headerSort']">
|
|||
|
|
{{ getSortIcon(col.key) }}
|
|||
|
|
</view>
|
|||
|
|
</template>
|
|||
|
|
</view>
|
|||
|
|
</view>
|
|||
|
|
</view>
|
|||
|
|
|
|||
|
|
<!-- 表体 -->
|
|||
|
|
<view class="u-table-body" :style="{ minWidth: scrollWidth, maxHeight: maxHeight ? maxHeight + 'px' : 'none' }">
|
|||
|
|
<template v-if="data && data.length > 0">
|
|||
|
|
<template v-for="(row, rowIndex) in sortedData" :key="row[rowKey] || rowIndex">
|
|||
|
|
<!-- 子级渲染 (递归组件) -->
|
|||
|
|
<table-row
|
|||
|
|
:row="row"
|
|||
|
|
:rowIndex="rowIndex"
|
|||
|
|
:parent-row="null"
|
|||
|
|
:columns="visibleFixedLeftColumns"
|
|||
|
|
: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="1"
|
|||
|
|
:rowHeight="rowHeight"
|
|||
|
|
:hasTree="hasTree"
|
|||
|
|
:selectedRows="selectedRows"
|
|||
|
|
:expandWidth="expandWidth"
|
|||
|
|
:computedMainCol="computedMainCol"
|
|||
|
|
:span-method="spanMethod"
|
|||
|
|
@toggle-select="toggleSelect"
|
|||
|
|
@row-click="handleRowClick"
|
|||
|
|
@toggle-expand="toggleExpand"
|
|||
|
|
>
|
|||
|
|
<template v-slot:cellChild="scope">
|
|||
|
|
<slot name="cell" :row="scope.row" :column="scope.column" :prow="scope.prow"
|
|||
|
|
:rowIndex="scope.rowIndex" :columnIndex="scope.columnIndex" :level="scope.level">
|
|||
|
|
</slot>
|
|||
|
|
</template>
|
|||
|
|
</table-row>
|
|||
|
|
</template>
|
|||
|
|
</template>
|
|||
|
|
</view>
|
|||
|
|
</view>
|
|||
|
|
</view>
|
|||
|
|
</template>
|
|||
|
|
|
|||
|
|
<script>
|
|||
|
|
import { addUnit, sleep } from '../../libs/function/index';
|
|||
|
|
import tableRow from './tableRow.vue'; // 引入递归组件
|
|||
|
|
|
|||
|
|
export default {
|
|||
|
|
name: 'u-table2',
|
|||
|
|
components: {
|
|||
|
|
tableRow // 注册递归组件
|
|||
|
|
},
|
|||
|
|
props: {
|
|||
|
|
data: {
|
|||
|
|
type: Array,
|
|||
|
|
required: true,
|
|||
|
|
default: () => {
|
|||
|
|
return []
|
|||
|
|
}
|
|||
|
|
},
|
|||
|
|
columns: {
|
|||
|
|
type: Array,
|
|||
|
|
required: true,
|
|||
|
|
default: () => {
|
|||
|
|
return []
|
|||
|
|
},
|
|||
|
|
validator: cols =>
|
|||
|
|
cols.every(col =>
|
|||
|
|
['default', 'selection', 'expand'].includes(col.type || 'default')
|
|||
|
|
)
|
|||
|
|
},
|
|||
|
|
stripe: {
|
|||
|
|
type: Boolean,
|
|||
|
|
default: false
|
|||
|
|
},
|
|||
|
|
border: {
|
|||
|
|
type: Boolean,
|
|||
|
|
default: false
|
|||
|
|
},
|
|||
|
|
height: {
|
|||
|
|
type: [String, Number],
|
|||
|
|
default: null
|
|||
|
|
},
|
|||
|
|
maxHeight: {
|
|||
|
|
type: [String, Number],
|
|||
|
|
default: null
|
|||
|
|
},
|
|||
|
|
showHeader: {
|
|||
|
|
type: Boolean,
|
|||
|
|
default: true
|
|||
|
|
},
|
|||
|
|
highlightCurrentRow: {
|
|||
|
|
type: Boolean,
|
|||
|
|
default: false
|
|||
|
|
},
|
|||
|
|
rowKey: {
|
|||
|
|
type: String,
|
|||
|
|
default: 'id'
|
|||
|
|
},
|
|||
|
|
currentRowKey: {
|
|||
|
|
type: [String, Number],
|
|||
|
|
default: null
|
|||
|
|
},
|
|||
|
|
rowStyle: {
|
|||
|
|
type: Object,
|
|||
|
|
default: () => ({})
|
|||
|
|
},
|
|||
|
|
cellClassName: {
|
|||
|
|
type: Function,
|
|||
|
|
default: null
|
|||
|
|
},
|
|||
|
|
cellStyle: {
|
|||
|
|
type: Function,
|
|||
|
|
default: null
|
|||
|
|
},
|
|||
|
|
headerCellClassName: {
|
|||
|
|
type: Function,
|
|||
|
|
default: null
|
|||
|
|
},
|
|||
|
|
rowClassName: {
|
|||
|
|
type: Function,
|
|||
|
|
default: null
|
|||
|
|
},
|
|||
|
|
context: {
|
|||
|
|
type: Object,
|
|||
|
|
default: null
|
|||
|
|
},
|
|||
|
|
showOverflowTooltip: {
|
|||
|
|
type: Boolean,
|
|||
|
|
default: false
|
|||
|
|
},
|
|||
|
|
lazy: {
|
|||
|
|
type: Boolean,
|
|||
|
|
default: false
|
|||
|
|
},
|
|||
|
|
load: {
|
|||
|
|
type: Function,
|
|||
|
|
default: null
|
|||
|
|
},
|
|||
|
|
treeProps: {
|
|||
|
|
type: Object,
|
|||
|
|
default: () => ({
|
|||
|
|
children: 'children',
|
|||
|
|
hasChildren: 'hasChildren'
|
|||
|
|
})
|
|||
|
|
},
|
|||
|
|
defaultExpandAll: {
|
|||
|
|
type: Boolean,
|
|||
|
|
default: false
|
|||
|
|
},
|
|||
|
|
expandRowKeys: {
|
|||
|
|
type: Array,
|
|||
|
|
default: () => []
|
|||
|
|
},
|
|||
|
|
sortOrders: {
|
|||
|
|
type: Array,
|
|||
|
|
default: () => ['ascending', 'descending']
|
|||
|
|
},
|
|||
|
|
sortable: {
|
|||
|
|
type: [Boolean, String],
|
|||
|
|
default: false
|
|||
|
|
},
|
|||
|
|
multiSort: {
|
|||
|
|
type: Boolean,
|
|||
|
|
default: false
|
|||
|
|
},
|
|||
|
|
sortBy: {
|
|||
|
|
type: String,
|
|||
|
|
default: null
|
|||
|
|
},
|
|||
|
|
sortMethod: {
|
|||
|
|
type: Function,
|
|||
|
|
default: null
|
|||
|
|
},
|
|||
|
|
filters: {
|
|||
|
|
type: Object,
|
|||
|
|
default: () => ({})
|
|||
|
|
},
|
|||
|
|
fixedHeader: {
|
|||
|
|
type: Boolean,
|
|||
|
|
default: true
|
|||
|
|
},
|
|||
|
|
emptyText: {
|
|||
|
|
type: String,
|
|||
|
|
default: '暂无数据'
|
|||
|
|
},
|
|||
|
|
// 添加mainCol属性,用于指定树形结构展开控制图标所在的列
|
|||
|
|
mainCol: {
|
|||
|
|
type: String,
|
|||
|
|
default: ''
|
|||
|
|
},
|
|||
|
|
expandWidth: {
|
|||
|
|
type: String,
|
|||
|
|
default: '25px'
|
|||
|
|
},
|
|||
|
|
rowHeight: {
|
|||
|
|
type: String,
|
|||
|
|
default: '36px'
|
|||
|
|
},
|
|||
|
|
// 添加spanMethod属性,用于合并单元格
|
|||
|
|
spanMethod: {
|
|||
|
|
type: Function,
|
|||
|
|
default: null
|
|||
|
|
}
|
|||
|
|
},
|
|||
|
|
emits: [
|
|||
|
|
'select', 'select-all', 'selection-change',
|
|||
|
|
'cell-click', 'row-click', 'row-dblclick',
|
|||
|
|
'header-click', 'sort-change', 'filter-change',
|
|||
|
|
'current-change', 'expand-change'
|
|||
|
|
],
|
|||
|
|
data() {
|
|||
|
|
return {
|
|||
|
|
scrollWidth: 'auto',
|
|||
|
|
// 将setup中的ref转换为data属性
|
|||
|
|
expandedKeys: [...this.expandRowKeys],
|
|||
|
|
selectedRows: [],
|
|||
|
|
sortConditions: [],
|
|||
|
|
currentRow: null,
|
|||
|
|
scrollLeft: 0, // 新增滚动位置数据
|
|||
|
|
showFixedColumnShadow: false, // 是否显示固定列阴影
|
|||
|
|
fixedLeftColumns: [], // 左侧固定列
|
|||
|
|
tableHeight: 'auto', // 表格高度
|
|||
|
|
headerHeight: 'auto', // 新增表头高度属性
|
|||
|
|
hasTree: false // 新增属性,用于判断是否存在树形结构
|
|||
|
|
}
|
|||
|
|
},
|
|||
|
|
mounted() {
|
|||
|
|
this.getComponentWidth()
|
|||
|
|
// 处理currentRowKey初始化
|
|||
|
|
if (this.currentRowKey !== null) {
|
|||
|
|
const found = this.data.find(item => item[this.rowKey] === this.currentRowKey);
|
|||
|
|
if (found) {
|
|||
|
|
this.currentRow = found;
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
// 获取固定列
|
|||
|
|
this.fixedLeftColumns = this.columns.filter(col => col.fixed === 'left');
|
|||
|
|
},
|
|||
|
|
computed: {
|
|||
|
|
// 将setup中的computed转换为computed属性
|
|||
|
|
filteredData() {
|
|||
|
|
return this.data.filter(row => {
|
|||
|
|
return Object.keys(this.filters).every(key => {
|
|||
|
|
const filter = this.filters[key];
|
|||
|
|
if (!filter) return true;
|
|||
|
|
return row[key]?.toString().includes(filter.toString());
|
|||
|
|
});
|
|||
|
|
});
|
|||
|
|
},
|
|||
|
|
sortedData() {
|
|||
|
|
if (!this.sortConditions.length) return this.filteredData;
|
|||
|
|
|
|||
|
|
const data = [...this.filteredData];
|
|||
|
|
|
|||
|
|
return data.sort((a, b) => {
|
|||
|
|
for (const condition of this.sortConditions) {
|
|||
|
|
const { field, order } = condition;
|
|||
|
|
let valA = a[field];
|
|||
|
|
let valB = b[field];
|
|||
|
|
|
|||
|
|
if (this.sortMethod) {
|
|||
|
|
const result = this.sortMethod(a, b, field);
|
|||
|
|
if (result !== 0) return result * (order === 'ascending' ? 1 : -1);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
if (valA < valB) return order === 'ascending' ? -1 : 1;
|
|||
|
|
if (valA > valB) return order === 'ascending' ? 1 : -1;
|
|||
|
|
}
|
|||
|
|
return 0;
|
|||
|
|
});
|
|||
|
|
},
|
|||
|
|
// 计算当前应该显示的固定左侧列
|
|||
|
|
visibleFixedLeftColumns() {
|
|||
|
|
if (this.scrollLeft <= 0) {
|
|||
|
|
return [];
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
let totalWidth = 0;
|
|||
|
|
let fixedWidth = 0;
|
|||
|
|
const visibleColumns = [];
|
|||
|
|
|
|||
|
|
// 遍历所有列,不仅仅是固定列
|
|||
|
|
for (let i = 0; i < this.columns.length; i++) {
|
|||
|
|
const col = this.columns[i];
|
|||
|
|
const colWidth = col.width ? parseInt(col.width) : 100; // 默认宽度100px
|
|||
|
|
|
|||
|
|
// 如果是固定列且滚动位置足够显示该列
|
|||
|
|
if (col.fixed === 'left' && this.scrollLeft > totalWidth - fixedWidth) {
|
|||
|
|
visibleColumns.push(col);
|
|||
|
|
fixedWidth += colWidth;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
totalWidth += colWidth;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
return visibleColumns;
|
|||
|
|
},
|
|||
|
|
// 获取mainCol的值,如果未设置则默认为第一列的key
|
|||
|
|
computedMainCol() {
|
|||
|
|
if (this.mainCol) {
|
|||
|
|
return this.mainCol;
|
|||
|
|
}
|
|||
|
|
// 修改为排除有type值的列
|
|||
|
|
const validColumns = this.columns.filter(col => !col.type);
|
|||
|
|
let mainCol = validColumns && validColumns.length > 0 ? validColumns[0].key : '';
|
|||
|
|
// console.log('mainCol', mainCol)
|
|||
|
|
return mainCol;
|
|||
|
|
}
|
|||
|
|
},
|
|||
|
|
watch: {
|
|||
|
|
// 将setup中的watch转换为watch属性
|
|||
|
|
expandRowKeys: {
|
|||
|
|
handler(newVal) {
|
|||
|
|
this.expandedKeys = [...newVal];
|
|||
|
|
},
|
|||
|
|
immediate: true
|
|||
|
|
},
|
|||
|
|
currentRowKey: {
|
|||
|
|
handler(newVal) {
|
|||
|
|
const found = this.data.find(item => item[this.rowKey] === newVal);
|
|||
|
|
if (found) {
|
|||
|
|
this.currentRow = found;
|
|||
|
|
}
|
|||
|
|
},
|
|||
|
|
immediate: true
|
|||
|
|
},
|
|||
|
|
columns: {
|
|||
|
|
handler() {
|
|||
|
|
// this.fixedLeftColumns = this.columns.filter(col => col.fixed === 'left');
|
|||
|
|
},
|
|||
|
|
deep: true,
|
|||
|
|
immediate: false
|
|||
|
|
}
|
|||
|
|
},
|
|||
|
|
methods: {
|
|||
|
|
addUnit,
|
|||
|
|
onScroll(e) {
|
|||
|
|
this.scrollLeft = e.detail.scrollLeft;
|
|||
|
|
// 获取所有左侧固定列
|
|||
|
|
this.fixedLeftColumns = this.columns.filter(col => col.fixed === 'left');
|
|||
|
|
// 计算是否需要显示固定列阴影
|
|||
|
|
if (this.fixedLeftColumns.length > 0) {
|
|||
|
|
this.showFixedColumnShadow = this.scrollLeft > 0;
|
|||
|
|
}
|
|||
|
|
},
|
|||
|
|
|
|||
|
|
getFixedShadowStyle(col, index) {
|
|||
|
|
let style = {
|
|||
|
|
width: col.width ? addUnit(col.width) : 'auto',
|
|||
|
|
};
|
|||
|
|
|
|||
|
|
if (col?.style) {
|
|||
|
|
style = {...style, ...col?.style};
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
return style;
|
|||
|
|
},
|
|||
|
|
|
|||
|
|
getFixedClass(col) {
|
|||
|
|
return ''; // 不再使用原来的固定列样式类
|
|||
|
|
},
|
|||
|
|
|
|||
|
|
headerColStyle(col) {
|
|||
|
|
let style = {
|
|||
|
|
width: col.width ? addUnit(col.width) : 'auto',
|
|||
|
|
flex: col.width ? 'none' : 1
|
|||
|
|
};
|
|||
|
|
if (col?.style) {
|
|||
|
|
style = {...style, ...col?.style};
|
|||
|
|
}
|
|||
|
|
return style;
|
|||
|
|
},
|
|||
|
|
|
|||
|
|
setCellStyle(e) {
|
|||
|
|
this.cellStyle = e
|
|||
|
|
},
|
|||
|
|
cellStyleInner(scope) {
|
|||
|
|
let style = {
|
|||
|
|
width: scope.column?.width ? addUnit(scope.column.width) : 'auto',
|
|||
|
|
flex: scope.column?.width ? 'none' : 1
|
|||
|
|
};
|
|||
|
|
// 只有展开列设置padding
|
|||
|
|
if (scope.column.key == this.computedMainCol) {
|
|||
|
|
style.paddingLeft = (16 * (scope.level -1 )) + 2 + 'px'
|
|||
|
|
}
|
|||
|
|
if (this.cellStyle != null) {
|
|||
|
|
let styleCalc = this.cellStyle(scope)
|
|||
|
|
if (styleCalc != null) {
|
|||
|
|
style = {...style, ...styleCalc}
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
return style;
|
|||
|
|
},
|
|||
|
|
// 获取组件的宽度
|
|||
|
|
async getComponentWidth() {
|
|||
|
|
// 延时一定时间,以获取dom尺寸
|
|||
|
|
await sleep(30)
|
|||
|
|
this.$uGetRect('.u-table-row').then(size => {
|
|||
|
|
this.scrollWidth = size.width + 'px'
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
// 获取表头高度并设置
|
|||
|
|
this.$uGetRect('.u-table-header').then(size => {
|
|||
|
|
if (size.height) {
|
|||
|
|
this.headerHeight = size.height + 'px';
|
|||
|
|
}
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
// 遍历数据列表第一层判断是否存在树形结构
|
|||
|
|
this.hasTree = this.sortedData.some(item => {
|
|||
|
|
return item[this.treeProps.children] && item[this.treeProps.children].length > 0;
|
|||
|
|
});
|
|||
|
|
},
|
|||
|
|
// 将setup中的函数转换为methods
|
|||
|
|
handleRowClick(row) {
|
|||
|
|
if (this.highlightCurrentRow) {
|
|||
|
|
const oldRow = this.currentRow;
|
|||
|
|
this.currentRow = row;
|
|||
|
|
this.$emit('current-change', row, oldRow);
|
|||
|
|
}
|
|||
|
|
this.$emit('row-click', row);
|
|||
|
|
},
|
|||
|
|
handleHeaderClick(column) {
|
|||
|
|
if (!column.sortable) return;
|
|||
|
|
|
|||
|
|
const index = this.sortConditions.findIndex(c => c.field === column.key);
|
|||
|
|
let newOrder = 'ascending';
|
|||
|
|
|
|||
|
|
if (index >= 0) {
|
|||
|
|
if (this.sortConditions[index].order === 'ascending') {
|
|||
|
|
newOrder = 'descending';
|
|||
|
|
} else {
|
|||
|
|
this.sortConditions.splice(index, 1);
|
|||
|
|
this.$emit('sort-change', this.sortConditions);
|
|||
|
|
return;
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
if (!this.multiSort) {
|
|||
|
|
this.sortConditions = [{ field: column.key, order: newOrder }];
|
|||
|
|
} else {
|
|||
|
|
if (index >= 0) {
|
|||
|
|
this.sortConditions[index].order = newOrder;
|
|||
|
|
} else {
|
|||
|
|
this.sortConditions.push({ field: column.key, order: newOrder });
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
this.$emit('sort-change', this.sortConditions);
|
|||
|
|
},
|
|||
|
|
getSortIcon(field) {
|
|||
|
|
const cond = this.sortConditions.find(c => c.field === field);
|
|||
|
|
if (!cond) return '';
|
|||
|
|
return cond.order === 'ascending' ? '↑' : '↓';
|
|||
|
|
},
|
|||
|
|
getSortValue(field) {
|
|||
|
|
const cond = this.sortConditions.find(c => c.field === field);
|
|||
|
|
if (!cond) return '';
|
|||
|
|
return cond.order === 'ascending';
|
|||
|
|
},
|
|||
|
|
toggleSelect(row) {
|
|||
|
|
const index = this.selectedRows.findIndex(r => r[this.rowKey] === row[this.rowKey]);
|
|||
|
|
if (index >= 0) {
|
|||
|
|
// 取消选中当前行及其所有子节点
|
|||
|
|
this.selectedRows.splice(index, 1);
|
|||
|
|
// 递归取消所有子节点
|
|||
|
|
this.unselectChildren(row);
|
|||
|
|
} else {
|
|||
|
|
// 选中当前行及其所有子节点
|
|||
|
|
this.selectedRows.push(row);
|
|||
|
|
// 递归选中所有子节点
|
|||
|
|
this.selectChildren(row);
|
|||
|
|
}
|
|||
|
|
console.log(this.selectedRows)
|
|||
|
|
this.$emit('selection-change', this.selectedRows);
|
|||
|
|
this.$emit('select', row);
|
|||
|
|
},
|
|||
|
|
toggleExpand(row) {
|
|||
|
|
// console.log(row)
|
|||
|
|
const key = row[this.rowKey];
|
|||
|
|
const index = this.expandedKeys.indexOf(key);
|
|||
|
|
if (index === -1) {
|
|||
|
|
this.expandedKeys.push(key);
|
|||
|
|
} else {
|
|||
|
|
this.expandedKeys.splice(index, 1);
|
|||
|
|
}
|
|||
|
|
this.$emit('expand-change', this.expandedKeys);
|
|||
|
|
},
|
|||
|
|
isExpanded(row) {
|
|||
|
|
if (!row) {
|
|||
|
|
return false;
|
|||
|
|
}
|
|||
|
|
return this.expandedKeys.includes(row[this.rowKey]);
|
|||
|
|
},
|
|||
|
|
// 新增方法:递归选中所有子节点
|
|||
|
|
selectChildren(row) {
|
|||
|
|
const children = row[this.treeProps.children];
|
|||
|
|
if (children && children.length > 0) {
|
|||
|
|
children.forEach(child => {
|
|||
|
|
// 检查是否已选中,避免重复添加
|
|||
|
|
const childIndex = this.selectedRows.findIndex(r => r[this.rowKey] === child[this.rowKey]);
|
|||
|
|
if (childIndex === -1) {
|
|||
|
|
this.selectedRows.push(child);
|
|||
|
|
}
|
|||
|
|
// 递归处理子节点的子节点
|
|||
|
|
this.selectChildren(child);
|
|||
|
|
});
|
|||
|
|
}
|
|||
|
|
},
|
|||
|
|
// 新增方法:递归取消选中所有子节点
|
|||
|
|
unselectChildren(row) {
|
|||
|
|
const children = row[this.treeProps.children];
|
|||
|
|
if (children && children.length > 0) {
|
|||
|
|
children.forEach(child => {
|
|||
|
|
const childIndex = this.selectedRows.findIndex(r => r[this.rowKey] === child[this.rowKey]);
|
|||
|
|
if (childIndex >= 0) {
|
|||
|
|
this.selectedRows.splice(childIndex, 1);
|
|||
|
|
}
|
|||
|
|
// 递归处理子节点的子节点
|
|||
|
|
this.unselectChildren(child);
|
|||
|
|
});
|
|||
|
|
}
|
|||
|
|
},
|
|||
|
|
}
|
|||
|
|
};
|
|||
|
|
</script>
|
|||
|
|
|
|||
|
|
<style lang="scss" scoped>
|
|||
|
|
.u-table2 {
|
|||
|
|
width: auto;
|
|||
|
|
overflow: auto;
|
|||
|
|
white-space: nowrap;
|
|||
|
|
position: relative;
|
|||
|
|
|
|||
|
|
.u-table-header {
|
|||
|
|
min-width: 100% !important;
|
|||
|
|
width: fit-content;
|
|||
|
|
background-color: #f5f7fa;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.u-table-body {
|
|||
|
|
min-width: 100% !important;
|
|||
|
|
width: fit-content;
|
|||
|
|
position: relative;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.u-table-sticky {
|
|||
|
|
position: sticky;
|
|||
|
|
top: 0;
|
|||
|
|
z-index: 10;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.u-table-row {
|
|||
|
|
display: flex;
|
|||
|
|
flex-direction: row;
|
|||
|
|
overflow: hidden;
|
|||
|
|
position: relative;
|
|||
|
|
// min-height: 40px;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 添加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-fixed-shadow {
|
|||
|
|
position: absolute;
|
|||
|
|
top: 0;
|
|||
|
|
left: 0;
|
|||
|
|
width: auto;
|
|||
|
|
z-index: 20;
|
|||
|
|
box-shadow: 2px 0 5px rgba(0, 0, 0, 0.15);
|
|||
|
|
overflow: hidden;
|
|||
|
|
background-color: #ffffff;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// .u-table-fixed-row {
|
|||
|
|
// display: flex;
|
|||
|
|
// flex-direction: row;
|
|||
|
|
// align-items: center;
|
|||
|
|
// border-bottom: 1rpx solid #ebeef5;
|
|||
|
|
// position: relative;
|
|||
|
|
// }
|
|||
|
|
|
|||
|
|
// 为固定列也添加border样式支持
|
|||
|
|
.u-table-fixed-shadow .u-table-border {
|
|||
|
|
.u-table-cell {
|
|||
|
|
border-right: 1rpx solid #ebeef5;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.u-table-cell:last-child {
|
|||
|
|
border-right: none;
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
</style>
|