456 lines
9.8 KiB
Vue
456 lines
9.8 KiB
Vue
|
<template>
|
|||
|
<view class="uni-table-scroll" :class="{ 'table--border': border, 'border-none': !noData }">
|
|||
|
<!-- #ifdef H5 -->
|
|||
|
<table class="uni-table" border="0" cellpadding="0" cellspacing="0" :class="{ 'table--stripe': stripe }" :style="{ 'min-width': minWidth + 'px' }">
|
|||
|
<slot></slot>
|
|||
|
<tr v-if="noData" class="uni-table-loading">
|
|||
|
<td class="uni-table-text" :class="{ 'empty-border': border }">{{ emptyText }}</td>
|
|||
|
</tr>
|
|||
|
<view v-if="loading" class="uni-table-mask" :class="{ 'empty-border': border }"><div class="uni-table--loader"></div></view>
|
|||
|
</table>
|
|||
|
<!-- #endif -->
|
|||
|
<!-- #ifndef H5 -->
|
|||
|
<view class="uni-table" :style="{ 'min-width': minWidth + 'px' }" :class="{ 'table--stripe': stripe }">
|
|||
|
<slot></slot>
|
|||
|
<view v-if="noData" class="uni-table-loading">
|
|||
|
<view class="uni-table-text" :class="{ 'empty-border': border }">{{ emptyText }}</view>
|
|||
|
</view>
|
|||
|
<view v-if="loading" class="uni-table-mask" :class="{ 'empty-border': border }"><div class="uni-table--loader"></div></view>
|
|||
|
</view>
|
|||
|
<!-- #endif -->
|
|||
|
</view>
|
|||
|
</template>
|
|||
|
|
|||
|
<script>
|
|||
|
/**
|
|||
|
* Table 表格
|
|||
|
* @description 用于展示多条结构类似的数据
|
|||
|
* @tutorial https://ext.dcloud.net.cn/plugin?id=3270
|
|||
|
* @property {Boolean} border 是否带有纵向边框
|
|||
|
* @property {Boolean} stripe 是否显示斑马线
|
|||
|
* @property {Boolean} type 是否开启多选
|
|||
|
* @property {String} emptyText 空数据时显示的文本内容
|
|||
|
* @property {Boolean} loading 显示加载中
|
|||
|
* @event {Function} selection-change 开启多选时,当选择项发生变化时会触发该事件
|
|||
|
*/
|
|||
|
export default {
|
|||
|
name: 'uniTable',
|
|||
|
options: {
|
|||
|
virtualHost: true
|
|||
|
},
|
|||
|
emits:['selection-change'],
|
|||
|
props: {
|
|||
|
data: {
|
|||
|
type: Array,
|
|||
|
default() {
|
|||
|
return []
|
|||
|
}
|
|||
|
},
|
|||
|
// 是否有竖线
|
|||
|
border: {
|
|||
|
type: Boolean,
|
|||
|
default: false
|
|||
|
},
|
|||
|
// 是否显示斑马线
|
|||
|
stripe: {
|
|||
|
type: Boolean,
|
|||
|
default: false
|
|||
|
},
|
|||
|
// 多选
|
|||
|
type: {
|
|||
|
type: String,
|
|||
|
default: ''
|
|||
|
},
|
|||
|
// 没有更多数据
|
|||
|
emptyText: {
|
|||
|
type: String,
|
|||
|
default: '没有更多数据'
|
|||
|
},
|
|||
|
loading: {
|
|||
|
type: Boolean,
|
|||
|
default: false
|
|||
|
},
|
|||
|
rowKey: {
|
|||
|
type: String,
|
|||
|
default: ''
|
|||
|
}
|
|||
|
},
|
|||
|
data() {
|
|||
|
return {
|
|||
|
noData: true,
|
|||
|
minWidth: 0,
|
|||
|
multiTableHeads: []
|
|||
|
}
|
|||
|
},
|
|||
|
watch: {
|
|||
|
loading(val) {},
|
|||
|
data(newVal) {
|
|||
|
let theadChildren = this.theadChildren
|
|||
|
let rowspan = 1
|
|||
|
if (this.theadChildren) {
|
|||
|
rowspan = this.theadChildren.rowspan
|
|||
|
}
|
|||
|
|
|||
|
// this.trChildren.length - rowspan
|
|||
|
this.noData = false
|
|||
|
// this.noData = newVal.length === 0
|
|||
|
}
|
|||
|
},
|
|||
|
created() {
|
|||
|
// 定义tr的实例数组
|
|||
|
this.trChildren = []
|
|||
|
this.thChildren = []
|
|||
|
this.theadChildren = null
|
|||
|
this.backData = []
|
|||
|
this.backIndexData = []
|
|||
|
},
|
|||
|
|
|||
|
methods: {
|
|||
|
isNodata() {
|
|||
|
let theadChildren = this.theadChildren
|
|||
|
let rowspan = 1
|
|||
|
if (this.theadChildren) {
|
|||
|
rowspan = this.theadChildren.rowspan
|
|||
|
}
|
|||
|
this.noData = this.trChildren.length - rowspan <= 0
|
|||
|
},
|
|||
|
/**
|
|||
|
* 选中所有
|
|||
|
*/
|
|||
|
selectionAll() {
|
|||
|
let startIndex = 1
|
|||
|
let theadChildren = this.theadChildren
|
|||
|
if (!this.theadChildren) {
|
|||
|
theadChildren = this.trChildren[0]
|
|||
|
} else {
|
|||
|
startIndex = theadChildren.rowspan - 1
|
|||
|
}
|
|||
|
let isHaveData = this.data && this.data.length > 0
|
|||
|
theadChildren.checked = true
|
|||
|
theadChildren.indeterminate = false
|
|||
|
this.trChildren.forEach((item, index) => {
|
|||
|
if (!item.disabled) {
|
|||
|
item.checked = true
|
|||
|
if (isHaveData && item.keyValue) {
|
|||
|
const row = this.data.find(v => v[this.rowKey] === item.keyValue)
|
|||
|
if (!this.backData.find(v => v[this.rowKey] === row[this.rowKey])) {
|
|||
|
this.backData.push(row)
|
|||
|
}
|
|||
|
}
|
|||
|
if (index > (startIndex - 1) && this.backIndexData.indexOf(index - startIndex) === -1) {
|
|||
|
this.backIndexData.push(index - startIndex)
|
|||
|
}
|
|||
|
}
|
|||
|
})
|
|||
|
// this.backData = JSON.parse(JSON.stringify(this.data))
|
|||
|
this.$emit('selection-change', {
|
|||
|
detail: {
|
|||
|
value: this.backData,
|
|||
|
index: this.backIndexData
|
|||
|
}
|
|||
|
})
|
|||
|
},
|
|||
|
/**
|
|||
|
* 用于多选表格,切换某一行的选中状态,如果使用了第二个参数,则是设置这一行选中与否(selected 为 true 则选中)
|
|||
|
*/
|
|||
|
toggleRowSelection(row, selected) {
|
|||
|
// if (!this.theadChildren) return
|
|||
|
row = [].concat(row)
|
|||
|
|
|||
|
this.trChildren.forEach((item, index) => {
|
|||
|
// if (item.keyValue) {
|
|||
|
|
|||
|
const select = row.findIndex(v => {
|
|||
|
//
|
|||
|
if (typeof v === 'number') {
|
|||
|
return v === index - 1
|
|||
|
} else {
|
|||
|
return v[this.rowKey] === item.keyValue
|
|||
|
}
|
|||
|
})
|
|||
|
let ischeck = item.checked
|
|||
|
if (select !== -1) {
|
|||
|
if (typeof selected === 'boolean') {
|
|||
|
item.checked = selected
|
|||
|
} else {
|
|||
|
item.checked = !item.checked
|
|||
|
}
|
|||
|
if (ischeck !== item.checked) {
|
|||
|
this.check(item.rowData||item, item.checked, item.rowData?item.keyValue:null, true)
|
|||
|
}
|
|||
|
}
|
|||
|
// }
|
|||
|
})
|
|||
|
this.$emit('selection-change', {
|
|||
|
detail: {
|
|||
|
value: this.backData,
|
|||
|
index:this.backIndexData
|
|||
|
}
|
|||
|
})
|
|||
|
},
|
|||
|
|
|||
|
/**
|
|||
|
* 用于多选表格,清空用户的选择
|
|||
|
*/
|
|||
|
clearSelection() {
|
|||
|
let theadChildren = this.theadChildren
|
|||
|
if (!this.theadChildren) {
|
|||
|
theadChildren = this.trChildren[0]
|
|||
|
}
|
|||
|
// if (!this.theadChildren) return
|
|||
|
theadChildren.checked = false
|
|||
|
theadChildren.indeterminate = false
|
|||
|
this.trChildren.forEach(item => {
|
|||
|
// if (item.keyValue) {
|
|||
|
item.checked = false
|
|||
|
// }
|
|||
|
})
|
|||
|
this.backData = []
|
|||
|
this.backIndexData = []
|
|||
|
this.$emit('selection-change', {
|
|||
|
detail: {
|
|||
|
value: [],
|
|||
|
index: []
|
|||
|
}
|
|||
|
})
|
|||
|
},
|
|||
|
/**
|
|||
|
* 用于多选表格,切换所有行的选中状态
|
|||
|
*/
|
|||
|
toggleAllSelection() {
|
|||
|
let list = []
|
|||
|
let startIndex = 1
|
|||
|
let theadChildren = this.theadChildren
|
|||
|
if (!this.theadChildren) {
|
|||
|
theadChildren = this.trChildren[0]
|
|||
|
} else {
|
|||
|
startIndex = theadChildren.rowspan - 1
|
|||
|
}
|
|||
|
this.trChildren.forEach((item, index) => {
|
|||
|
if (!item.disabled) {
|
|||
|
if (index > (startIndex - 1) ) {
|
|||
|
list.push(index-startIndex)
|
|||
|
}
|
|||
|
}
|
|||
|
})
|
|||
|
this.toggleRowSelection(list)
|
|||
|
},
|
|||
|
|
|||
|
/**
|
|||
|
* 选中\取消选中
|
|||
|
* @param {Object} child
|
|||
|
* @param {Object} check
|
|||
|
* @param {Object} rowValue
|
|||
|
*/
|
|||
|
check(child, check, keyValue, emit) {
|
|||
|
let theadChildren = this.theadChildren
|
|||
|
if (!this.theadChildren) {
|
|||
|
theadChildren = this.trChildren[0]
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
let childDomIndex = this.trChildren.findIndex((item, index) => child === item)
|
|||
|
if(childDomIndex < 0){
|
|||
|
childDomIndex = this.data.findIndex(v=>v[this.rowKey] === keyValue) + 1
|
|||
|
}
|
|||
|
const dataLen = this.trChildren.filter(v => !v.disabled && v.keyValue).length
|
|||
|
if (childDomIndex === 0) {
|
|||
|
check ? this.selectionAll() : this.clearSelection()
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
if (check) {
|
|||
|
if (keyValue) {
|
|||
|
this.backData.push(child)
|
|||
|
}
|
|||
|
this.backIndexData.push(childDomIndex - 1)
|
|||
|
} else {
|
|||
|
const index = this.backData.findIndex(v => v[this.rowKey] === keyValue)
|
|||
|
const idx = this.backIndexData.findIndex(item => item === childDomIndex - 1)
|
|||
|
if (keyValue) {
|
|||
|
this.backData.splice(index, 1)
|
|||
|
}
|
|||
|
this.backIndexData.splice(idx, 1)
|
|||
|
}
|
|||
|
|
|||
|
const domCheckAll = this.trChildren.find((item, index) => index > 0 && !item.checked && !item.disabled)
|
|||
|
if (!domCheckAll) {
|
|||
|
theadChildren.indeterminate = false
|
|||
|
theadChildren.checked = true
|
|||
|
} else {
|
|||
|
theadChildren.indeterminate = true
|
|||
|
theadChildren.checked = false
|
|||
|
}
|
|||
|
|
|||
|
if (this.backIndexData.length === 0) {
|
|||
|
theadChildren.indeterminate = false
|
|||
|
}
|
|||
|
|
|||
|
if (!emit) {
|
|||
|
this.$emit('selection-change', {
|
|||
|
detail: {
|
|||
|
value: this.backData,
|
|||
|
index: this.backIndexData
|
|||
|
}
|
|||
|
})
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
</script>
|
|||
|
|
|||
|
<style lang="scss">
|
|||
|
$border-color: #ebeef5;
|
|||
|
|
|||
|
.uni-table-scroll {
|
|||
|
width: 100%;
|
|||
|
/* #ifndef APP-NVUE */
|
|||
|
overflow-x: auto;
|
|||
|
/* #endif */
|
|||
|
}
|
|||
|
|
|||
|
.uni-table {
|
|||
|
position: relative;
|
|||
|
width: 100%;
|
|||
|
border-radius: 5px;
|
|||
|
// box-shadow: 0px 0px 3px 1px rgba(0, 0, 0, 0.1);
|
|||
|
background-color: #fff;
|
|||
|
/* #ifndef APP-NVUE */
|
|||
|
box-sizing: border-box;
|
|||
|
display: table;
|
|||
|
overflow-x: auto;
|
|||
|
::v-deep .uni-table-tr:nth-child(n + 2) {
|
|||
|
&:hover {
|
|||
|
background-color: #f5f7fa;
|
|||
|
}
|
|||
|
}
|
|||
|
::v-deep .uni-table-thead {
|
|||
|
.uni-table-tr {
|
|||
|
// background-color: #f5f7fa;
|
|||
|
&:hover {
|
|||
|
background-color:#fafafa;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
/* #endif */
|
|||
|
}
|
|||
|
|
|||
|
.table--border {
|
|||
|
border: 1px $border-color solid;
|
|||
|
border-right: none;
|
|||
|
}
|
|||
|
|
|||
|
.border-none {
|
|||
|
/* #ifndef APP-NVUE */
|
|||
|
border-bottom: none;
|
|||
|
/* #endif */
|
|||
|
}
|
|||
|
|
|||
|
.table--stripe {
|
|||
|
/* #ifndef APP-NVUE */
|
|||
|
::v-deep .uni-table-tr:nth-child(2n + 3) {
|
|||
|
background-color: #fafafa;
|
|||
|
}
|
|||
|
/* #endif */
|
|||
|
}
|
|||
|
|
|||
|
/* 表格加载、无数据样式 */
|
|||
|
.uni-table-loading {
|
|||
|
position: relative;
|
|||
|
/* #ifndef APP-NVUE */
|
|||
|
display: table-row;
|
|||
|
/* #endif */
|
|||
|
height: 50px;
|
|||
|
line-height: 50px;
|
|||
|
overflow: hidden;
|
|||
|
box-sizing: border-box;
|
|||
|
}
|
|||
|
.empty-border {
|
|||
|
border-right: 1px $border-color solid;
|
|||
|
}
|
|||
|
.uni-table-text {
|
|||
|
position: absolute;
|
|||
|
right: 0;
|
|||
|
left: 0;
|
|||
|
text-align: center;
|
|||
|
font-size: 14px;
|
|||
|
color: #999;
|
|||
|
}
|
|||
|
|
|||
|
.uni-table-mask {
|
|||
|
position: absolute;
|
|||
|
top: 0;
|
|||
|
bottom: 0;
|
|||
|
left: 0;
|
|||
|
right: 0;
|
|||
|
background-color: rgba(255, 255, 255, 0.8);
|
|||
|
z-index: 99;
|
|||
|
/* #ifndef APP-NVUE */
|
|||
|
display: flex;
|
|||
|
margin: auto;
|
|||
|
transition: all 0.5s;
|
|||
|
/* #endif */
|
|||
|
justify-content: center;
|
|||
|
align-items: center;
|
|||
|
}
|
|||
|
|
|||
|
.uni-table--loader {
|
|||
|
width: 30px;
|
|||
|
height: 30px;
|
|||
|
border: 2px solid #aaa;
|
|||
|
// border-bottom-color: transparent;
|
|||
|
border-radius: 50%;
|
|||
|
/* #ifndef APP-NVUE */
|
|||
|
animation: 2s uni-table--loader linear infinite;
|
|||
|
/* #endif */
|
|||
|
position: relative;
|
|||
|
}
|
|||
|
|
|||
|
@keyframes uni-table--loader {
|
|||
|
0% {
|
|||
|
transform: rotate(360deg);
|
|||
|
}
|
|||
|
|
|||
|
10% {
|
|||
|
border-left-color: transparent;
|
|||
|
}
|
|||
|
|
|||
|
20% {
|
|||
|
border-bottom-color: transparent;
|
|||
|
}
|
|||
|
|
|||
|
30% {
|
|||
|
border-right-color: transparent;
|
|||
|
}
|
|||
|
|
|||
|
40% {
|
|||
|
border-top-color: transparent;
|
|||
|
}
|
|||
|
|
|||
|
50% {
|
|||
|
transform: rotate(0deg);
|
|||
|
}
|
|||
|
|
|||
|
60% {
|
|||
|
border-top-color: transparent;
|
|||
|
}
|
|||
|
|
|||
|
70% {
|
|||
|
border-left-color: transparent;
|
|||
|
}
|
|||
|
|
|||
|
80% {
|
|||
|
border-bottom-color: transparent;
|
|||
|
}
|
|||
|
|
|||
|
90% {
|
|||
|
border-right-color: transparent;
|
|||
|
}
|
|||
|
|
|||
|
100% {
|
|||
|
transform: rotate(-360deg);
|
|||
|
}
|
|||
|
}
|
|||
|
</style>
|