el-table自适应高度指令
v-el-table-auto-height 表格自动高度
原理:监听浏览器窗体变化、vue update渲染,取容器顶部和整个窗体高度计算容器高度
动态计算el-table的高度,使其底部与页面底部距离保持在一定的数值,包括:初始化动态计算、窗口变化动态计算、宽度变化动态计算(宽度变化可能导致一些元素折行)
- 参数说明:v-el-table-auto-height.noPager:demo="0"
- .noPager:无分页器时,默认分页器高度28px
- :demo:自定义容器时的选择器,cless不写点“.”,id要写井号“#”
- ="0",自定义底部距离,默认底部距离12px
stackblitz
演示
使用方法
// 在main.js引入
import elTableAutoHeight from '@/directive/el-table-auto-height/el-table-auto-height
Vue.use(elTableAutoHeight)
<!-- 在表格中使用,默认距离底部12px,默认分页器高度28px已扣除 -->
<el-table
v-el-table-auto-height
:data="dataList"
>
<el-table-column prop="demo" label="demo" />
</el-table>
<!-- 分页器 -->
<div class="block" style="float: right">
<el-pagination />
</div>
<!-- 自定义底部距离,设置0也存在扣除28px分页器高度 -->
<el-table
v-el-table-auto-height="0"
:data="dataList"
>
<el-table-column prop="demo" label="demo" />
</el-table>
<!-- 自定义选择器,由于vue语法问题,类选择器不写. id选择器需要写# -->
<!-- 没有分页器时加上.noPager -->
<div class="demo" v-el-table-auto-height.noPager:demo></div>
<div id="demo" v-el-table-auto-height.noPager:#demo></div>
源码
// 过渡动画
const TRANSITION_DURATION = '300ms'
// 指令内部变量命名空间
const NAMESPACES = '_elTableAutoHeight_namespaces'
/**
* el-table自动高度
* @param Vue
*/
const install = (Vue) => {
Vue.directive('el-table-auto-height', {
bind(el, binding, node) {
// 指令的局限性导致要打破数据单项传递的规则
// 删除代理
delete node.componentInstance.$props.height
// 重新设置代理
Vue.util.defineReactive(node.componentInstance, 'height')
init(el, binding, node)
doWork(el, binding, node)
},
update(el, binding, node) {
// 检测机制优化 TODO
if (getContext(el, binding, node).setTimout) {
clearTimeout(getContext(el, binding, node).setTimout)
}
getContext(el, binding, node).setTimout = setTimeout(() => {
doWork(el, binding, node, false)
}, 100)
},
unbind(el, binding, node) {
if (!node) {
return
}
if (getContext(el, binding, node).setTimout) {
clearTimeout(getContext(el, binding, node).setTimout)
}
delete node.context[NAMESPACES][getContextKey(el, binding, node)]
}
})
}
const init = (el, binding, node) => {
if (!node.context[NAMESPACES]) {
node.context[NAMESPACES] = {}
}
if (getContext(el, binding, node)) {
return
}
node.context[NAMESPACES][getContextKey(el, binding, node)] = {
id: Math.random().toString(36).slice(-8),
// 防抖器
setTimout: null,
container: null,
bottom: null,
node: node,
el: el,
binding: binding,
tableBody: null
}
}
const getContext = (el, binding, node) => {
const key = getContextKey(el, binding, node)
return node.context[NAMESPACES][key]
}
const getContextKey = (el, binding, node) => {
return `${node.tag}-${el.id}-${el.class}-${binding.arg}`
}
/**
* 处理容器、参数
* @param el
* @param binding
* @param node
* @param observeFlag 是否创建监听
*/
const doWork = (el, binding, node, observeFlag = true) => {
const container = binding.arg ? el.querySelector(getSelector(binding)) : el
// 默认距离底部12px
let bottom = binding.value
if (!bottom && bottom !== 0) {
bottom = 12
}
// 分页器高度
if (!binding.modifiers.noPager) {
bottom += 28
}
const context = getContext(el, binding, node)
context.bottom = bottom
context.container = container
context.tableBody = container.querySelector('.el-table__body-wrapper')
setHeight(context)
if (observeFlag) {
observeContainerChange(context)
}
}
/**
* 监听容器变化,由于el-table初始化渲染会导致宽度变化,所以初始化时会被调用多次
* @param context
*/
const observeContainerChange = (context) => {
// 宽度变化监听由于el-table内部计算会导致死循环,改为vue渲染变化监听 TODO
// const config = { attributes: true, childList: true, subtree: true, attributeFilter: ['width'] }
// const callback = (mutationsList, observer) => {
// setHeight(container, bottom, node, elTableFlag)
// }
// const observer = new MutationObserver(callback)
// observer.observe(container, config)
// 窗口变化监听
window.addEventListener('resize', () => {
setHeight(context)
})
}
/**
* 设置高度
* @param context
*/
const setHeight = (context) => {
const top = context.container.getBoundingClientRect().top
const innerHeight = window.innerHeight
let height = innerHeight - top - context.bottom
const body = context.container.querySelector('.el-table__body-wrapper')
if (body) {
context.container.style.transitionDuration = TRANSITION_DURATION
const footer = context.container.querySelector('.el-table__footer-wrapper')
if (footer) {
height = height - Number(footer.offsetHeight)
}
context.node.componentInstance.height = height
// 调用内部计算方法
context.node.componentInstance.layout.setHeight(height)
} else {
context.container.style.height = `${height}px`
context.container.style.overflowY = 'auto'
context.container.style.transitionDuration = TRANSITION_DURATION
}
}
/**
* 获取当前选择器
* @param binding
* @returns {string|string}
*/
const getSelector = (binding) => {
let containerSelector = binding.arg
if (containerSelector && containerSelector[0] !== '#' && containerSelector[0] !== '.') {
containerSelector = '.' + containerSelector
}
return containerSelector || 'default'
}
export default install