通过属性透传,iview Table组件实现自适应高度

前言

最近产品大大找我一顿吐槽,说我们的后台管理系统表格数据超出屏幕时,都得滑到最下面才能去拉横向滚动条,体验非常不好。

个人觉得键盘左右键也挺好的,要不再忍忍?

行,听产品的

实现思路

由于项目中涉及的页面过多,一个个页面修改肯定不现实。

可以利用vue的属性透传,在不改变原Table组件功能的前提下,包装原组件,新增自适应高度方法,列表页需要的就替换成新组件,其它页则不受影响。

属性透传涉及,属性,方法以及作用域插槽。

ref属性并不支持透传,为了方便原组件事件调用,可以在新组件中,对原组件方法进行代理访问.

实现效果图

具体实现

新建AutoHeightTable组件,进行属性透传

通过Vue提供的h函数方法,进行属性,方法和作用域插槽的透传。

$attrs为属性,$listeners为方法,$scopedSlots保存作用域插槽。

js 复制代码
import { h } from 'vue'

render() {
    return h('Table', {
        ref: 'table',
        props: {
            ...this.$attrs,
            // 传入计算后的最大高度
            height: this.maxHeight
        },
        on: this.$listeners,
        scopedSlots: this.$scopedSlots
    })
}

通过MutationObserver监听dom变化

通过MutationObserver监听表格dom的变化,dom变化后,计算当前页面的剩余空间,并将值赋予height属性。

用户缩小屏幕尺寸时,表格顶部的搜索框会换行,这个时候就需要重新计算表格高度。

由于产品要求默认充满屏幕,滚动条始终在底部,所以使用的height属性,个人觉得maxHeight会更加友好。
由于要保持页面源码的最小改动,默认底部距离bottom给了固定页脚高度80,如需对页脚高度进行计算,可通过传入类名或元素dom的方式进行计算。

js 复制代码
mounted() {
    this.initObServer()
},
beforeDestroy() {
    this.disconnectObserver()
},
methods: {
    initObServer() {
        if (!this.$refs.table) {
            return
        }
        // 通过MutationObserver监听dom变化
        this.observer = new MutationObserver(() => {
            this.getMaxHeight()
        })
        this.observer.observe(this.$refs.table.$el, {
            childList: true
        })
        // 监听尺寸变化
        window.addEventListener('resize', this.getMaxHeight)
    },
    disconnectObserver() {
        // 关闭监听
        if (this.observer) {
            this.observer.disconnect()
            this.observer = null
        }
        window.removeEventListener('resize', this.getMaxHeight)
    },
    // 计算高度
    getMaxHeight() {
        if (!this.$refs.table) {
            return
        }
        const { top } = this.$refs.table.$el.getBoundingClientRect()
        // 屏幕剩余空间 = 屏幕高度 - 表格顶部 - 表格底部
        this.maxHeight = window.innerHeight - top - this.bottom
    }
}

原Table组件方法代理

包装原组件经常会遇到一个问题

父组件通过ref直接调用组件内部的方法,而vue中并没有如react进行ref透传的方法。

父组件调用方法则必须通过refs.selection.refs.table.getSelection()。

html 复制代码
<AutoHeightTable
    border
    :loading="loading"
    :columns="columnsData"
    :data="tableData"
    ref="selection"></AutoHeightTable>
js 复制代码
// 获取表格选中项
this.$refs.selection.$refs.table.getSelection()

为了最小化改动源代码,通过属性代理的方式,来减少ref链的调用。

js 复制代码
// 代理属性
const PROXY_PROPERTIES = [
    'getSelection'
]

beforeCreate() {
    // ref无法透传,对相关属性进行代理
    PROXY_PROPERTIES.forEach(key => {
        Object.defineProperty(this, key, {
            get() {
                return this.$refs.table[key]
            }
        })
    })
}

此时源码中的调用便会正常获取到

js 复制代码
// 获取表格选中项
this.$refs.selection.getSelection()

完整代码

js 复制代码
import { h } from 'vue'

// 代理属性
const PROXY_PROPERTIES = [
    'getSelection'
]

export default {
    name: 'auto-table',
    props: {
        // 底部距离
        bottom: {
            type: Number,
            default: 80
        }
    },
    data() {
        return {
            maxHeight: 0
        }
    },
    beforeCreate() {
        // ref无法透传,对相关属性进行代理
        PROXY_PROPERTIES.forEach(key => {
            Object.defineProperty(this, key, {
                get() {
                    return this.$refs.table[key]
                }
            })
        })
    },
    mounted() {
        this.initObServer()
    },
    beforeDestroy() {
        this.disconnectObserver()
    },
    methods: {
        initObServer() {
            if (!this.$refs.table) {
                return
            }
            this.observer = new MutationObserver(() => {
                this.getMaxHeight()
            })
            this.observer.observe(this.$refs.table.$el, {
                childList: true
            })
            window.addEventListener('resize', this.getMaxHeight)
        },
        disconnectObserver() {
            if (this.observer) {
                this.observer.disconnect()
                this.observer = null
            }
            window.removeEventListener('resize', this.getMaxHeight)
        },
        getMaxHeight() {
            if (!this.$refs.table) {
                return
            }
            const { top } = this.$refs.table.$el.getBoundingClientRect()
            this.maxHeight = window.innerHeight - top - this.bottom
        }
    },
    render() {
        return h('Table', {
            ref: 'table',
            props: {
                ...this.$attrs,
                height: this.maxHeight
            },
            on: this.$listeners,
            scopedSlots: this.$scopedSlots
        })
    }
}
相关推荐
热忱11281 小时前
elementUI Table组件实现表头吸顶效果
前端·vue.js·elementui
w(゚Д゚)w吓洗宝宝了2 小时前
单例模式 - 单例模式的实现与应用
开发语言·javascript·单例模式
大叔_爱编程2 小时前
wx035基于springboot+vue+uniapp的校园二手交易小程序
vue.js·spring boot·小程序·uni-app·毕业设计·源码·课程设计
zhaocarbon2 小时前
VUE elTree 无子级 隐藏展开图标
前端·javascript·vue.js
小周不摆烂3 小时前
探索JavaScript前端开发:开启交互之门的神奇钥匙(二)
javascript
匹马夕阳4 小时前
Vue 3中导航守卫(Navigation Guard)结合Axios实现token认证机制
前端·javascript·vue.js
你熬夜了吗?4 小时前
日历热力图,月度数据可视化图表(日活跃图、格子图)vue组件
前端·vue.js·信息可视化
我想学LINUX5 小时前
【2024年华为OD机试】 (A卷,100分)- 微服务的集成测试(JavaScript&Java & Python&C/C++)
java·c语言·javascript·python·华为od·微服务·集成测试
screct_demo5 小时前
詳細講一下在RN(ReactNative)中,6個比較常用的組件以及詳細的用法
javascript·react native·react.js
CodeClimb12 小时前
【华为OD-E卷 - 第k个排列 100分(python、java、c++、js、c)】
java·javascript·c++·python·华为od