element-plus/packages/table/src/table-layout.ts

329 lines
9.8 KiB
TypeScript
Raw Normal View History

import { nextTick, ref, isRef, Ref } from 'vue'
import scrollbarWidth from '@element-plus/utils/scrollbar-width'
import isServer from '@element-plus/utils/isServer'
import { parseHeight } from './util'
import {
AnyObject,
Table,
Store,
TableHeader,
TableColumnCtx,
} from './table.type'
class TableLayout {
observers: TableHeader[]
table: Table
store: Store
columns: TableColumnCtx[]
fit: boolean
showHeader: boolean
height: Ref<null | number>
scrollX: Ref<boolean>
scrollY: Ref<boolean>
bodyWidth: Ref<null | number>
fixedWidth: Ref<null | number>
rightFixedWidth: Ref<null | number>
tableHeight: Ref<null | number>
headerHeight: Ref<null | number> // Table Header Height
appendHeight: Ref<null | number> // Append Slot Height
footerHeight: Ref<null | number> // Table Footer Height
viewportHeight: Ref<null | number> // Table Height - Scroll Bar Height
bodyHeight: Ref<null | number> // Table Height - Table Header Height
fixedBodyHeight: Ref<null | number> // Table Height - Table Header Height - Scroll Bar Height
gutterWidth: number
constructor (options: AnyObject) {
this.observers = []
this.table = null
this.store = null
this.columns = []
this.fit = true
this.showHeader = true
this.height = ref(null)
this.scrollX = ref(false)
this.scrollY = ref(false)
this.bodyWidth = ref(null)
this.fixedWidth = ref(null)
this.rightFixedWidth = ref(null)
this.tableHeight = ref(null)
this.headerHeight = ref(44)
this.appendHeight = ref(0)
this.footerHeight = ref(44)
this.viewportHeight = ref(null)
this.bodyHeight = ref(null)
this.fixedBodyHeight = ref(null)
this.gutterWidth = scrollbarWidth()
for (const name in options) {
if (options.hasOwnProperty(name)) {
if (isRef(this[name])) {
this[name].value = options[name]
} else {
this[name] = options[name]
}
}
}
if (!this.table) {
throw new Error('table is required for Table Layout')
}
if (!this.store) {
throw new Error('store is required for Table Layout')
}
}
updateScrollY () {
const height = this.height.value
/**
* When the height is not initialized, it is null.
* After the table is initialized, when the height is not configured, the height is 0.
*/
if (height === null) return false
const bodyWrapper = this.table.refs.bodyWrapper as HTMLElement
if (this.table.vnode.el && bodyWrapper) {
let scrollY = true
const prevScrollY = this.scrollY.value
/**
* When bodyHeight has no value,
* it means that the table height is not set,
* and the scroll bar will never appear
*/
if (this.bodyHeight.value === null) {
scrollY = false
} else {
const body = bodyWrapper.querySelector('.el-table__body') as HTMLElement
scrollY = body.offsetHeight > this.bodyHeight.value
}
this.scrollY.value = scrollY
return prevScrollY !== scrollY
}
return false
}
setHeight (value: string | number, prop = 'height') {
if (isServer) return
const el = this.table.vnode.el
value = parseHeight(value)
this.height.value = Number(value)
if (!el && (value || value === 0))
return nextTick(() => this.setHeight(value, prop))
if (typeof value === 'number') {
el.style[prop] = value + 'px'
this.updateElsHeight()
} else if (typeof value === 'string') {
el.style[prop] = value
this.updateElsHeight()
}
}
setMaxHeight (value: string | number) {
this.setHeight(value, 'max-height')
}
getFlattenColumns (): TableColumnCtx[] {
const flattenColumns = []
const columns = this.table.store.states.columns.value
columns.forEach(column => {
if (column.isColumnGroup) {
// eslint-disable-next-line prefer-spread
flattenColumns.push.apply(flattenColumns, column.columns)
} else {
flattenColumns.push(column)
}
})
return flattenColumns
}
updateElsHeight () {
if (!this.table.$ready) return nextTick(() => this.updateElsHeight())
const {
headerWrapper: headerWrapper_,
appendWrapper: appendWrapper_,
footerWrapper: footerWrapper_,
} = this.table.refs
const appendWrapper = appendWrapper_ as HTMLElement
const headerWrapper = headerWrapper_ as HTMLElement
const footerWrapper = footerWrapper_ as HTMLElement
this.appendHeight.value = appendWrapper ? appendWrapper.offsetHeight : 0
if (this.showHeader && !headerWrapper) return
// fix issue (https://github.com/ElemeFE/element/pull/16956)
const headerTrElm = headerWrapper
? headerWrapper.querySelector('.el-table__header tr')
: null
const noneHeader = this.headerDisplayNone(headerTrElm as HTMLElement)
const headerHeight = (this.headerHeight.value = !this.showHeader
? 0
: headerWrapper.offsetHeight)
if (
this.showHeader &&
!noneHeader &&
headerWrapper.offsetWidth > 0 &&
(this.table.store.states.columns.value || []).length > 0 &&
headerHeight < 2
) {
return nextTick(() => this.updateElsHeight())
}
const tableHeight = (this.tableHeight.value = this.table.vnode.el.clientHeight)
const footerHeight = (this.footerHeight.value = footerWrapper
? footerWrapper.offsetHeight
: 0)
if (this.height.value !== null) {
this.bodyHeight.value =
tableHeight - headerHeight - footerHeight + (footerWrapper ? 1 : 0)
}
this.fixedBodyHeight.value = this.scrollX.value
? this.bodyHeight.value - this.gutterWidth
: this.bodyHeight.value
this.viewportHeight.value = this.scrollX.value
? tableHeight - this.gutterWidth
: tableHeight
this.updateScrollY()
this.notifyObservers('scrollable')
}
headerDisplayNone (elm: HTMLElement) {
if (!elm) return true
let headerChild = elm
while (headerChild.tagName !== 'DIV') {
if (getComputedStyle(headerChild).display === 'none') {
return true
}
headerChild = headerChild.parentElement
}
return false
}
updateColumnsWidth () {
if (isServer) return
const fit = this.fit
const bodyWidth = this.table.vnode.el.clientWidth
let bodyMinWidth = 0
const flattenColumns = this.getFlattenColumns()
const flexColumns = flattenColumns.filter(
column => typeof column.width !== 'number',
)
flattenColumns.forEach(column => {
// Clean those columns whose width changed from flex to unflex
if (typeof column.width === 'number' && column.realWidth)
column.realWidth = null
})
if (flexColumns.length > 0 && fit) {
flattenColumns.forEach(column => {
bodyMinWidth += column.width || column.minWidth || 80
})
const scrollYWidth = this.scrollY.value ? this.gutterWidth : 0
if (bodyMinWidth <= bodyWidth - scrollYWidth) {
// DON'T HAVE SCROLL BAR
this.scrollX.value = false
const totalFlexWidth = bodyWidth - scrollYWidth - bodyMinWidth
if (flexColumns.length === 1) {
flexColumns[0].realWidth =
(flexColumns[0].minWidth || 80) + totalFlexWidth
} else {
const allColumnsWidth = flexColumns.reduce(
(prev, column) => prev + (column.minWidth || 80),
0,
)
const flexWidthPerPixel = totalFlexWidth / allColumnsWidth
let noneFirstWidth = 0
flexColumns.forEach((column, index) => {
if (index === 0) return
const flexWidth = Math.floor(
(column.minWidth || 80) * flexWidthPerPixel,
)
noneFirstWidth += flexWidth
column.realWidth = (column.minWidth || 80) + flexWidth
})
flexColumns[0].realWidth =
(flexColumns[0].minWidth || 80) + totalFlexWidth - noneFirstWidth
}
} else {
// HAVE HORIZONTAL SCROLL BAR
this.scrollX.value = true
flexColumns.forEach(function (column) {
column.realWidth = column.minWidth
})
}
this.bodyWidth.value = Math.max(bodyMinWidth, bodyWidth)
this.table.state.resizeState.value.width = this.bodyWidth.value
} else {
flattenColumns.forEach(column => {
if (!column.width && !column.minWidth) {
column.realWidth = 80
} else {
column.realWidth = column.width || column.minWidth
}
bodyMinWidth += column.realWidth
})
this.scrollX.value = bodyMinWidth > bodyWidth
this.bodyWidth.value = bodyMinWidth
}
const fixedColumns = this.store.states.fixedColumns.value
if (fixedColumns.length > 0) {
let fixedWidth = 0
fixedColumns.forEach(function (column) {
fixedWidth += column.realWidth || column.width
})
this.fixedWidth.value = fixedWidth
}
const rightFixedColumns = this.store.states.rightFixedColumns.value
if (rightFixedColumns.length > 0) {
let rightFixedWidth = 0
rightFixedColumns.forEach(function (column) {
rightFixedWidth += column.realWidth || column.width
})
this.rightFixedWidth.value = rightFixedWidth
}
this.notifyObservers('columns')
}
addObserver (observer: TableHeader) {
this.observers.push(observer)
}
removeObserver (observer: TableHeader) {
const index = this.observers.indexOf(observer)
if (index !== -1) {
this.observers.splice(index, 1)
}
}
notifyObservers (event: string) {
const observers = this.observers
observers.forEach(observer => {
switch (event) {
case 'columns':
observer.state?.onColumnsChange(this)
break
case 'scrollable':
observer.state?.onScrollableChange(this)
break
default:
throw new Error(`Table Layout don't have event ${event}.`)
}
})
}
}
export default TableLayout