element-plus/packages/components/table/src/table-layout.ts
三咲智子 55348b30b6
style: use prettier (#3228)
* style: use prettier

* style: just prettier format, no code changes

* style: eslint fix
object-shorthand, prefer-const

* style: fix no-void

* style: no-console
2021-09-04 19:29:28 +08:00

322 lines
9.8 KiB
TypeScript

import { nextTick, ref, isRef } from 'vue'
import { hasOwn } from '@vue/shared'
import scrollbarWidth from '@element-plus/utils/scrollbar-width'
import isServer from '@element-plus/utils/isServer'
import { parseHeight } from './util'
import { Ref } from 'vue'
import type { TableColumnCtx } from './table-column/defaults'
import type { TableHeader } from './table-header'
import type { Table } from './table/defaults'
import type { Store } from './store'
class TableLayout<T> {
observers: TableHeader[]
table: Table<T>
store: Store<T>
columns: TableColumnCtx<T>[]
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: Record<string, any>) {
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 (hasOwn(options, name)) {
if (isRef(this[name])) {
this[name as string].value = options[name]
} else {
this[name as string] = 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<T>[] {
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, appendWrapper, footerWrapper } = this.table.refs
this.appendHeight.value = appendWrapper ? appendWrapper.offsetHeight : 0
if (this.showHeader && !headerWrapper) return
const headerTrElm: HTMLElement = headerWrapper
? headerWrapper.querySelector('.el-table__header tr')
: null
const noneHeader = this.headerDisplayNone(headerTrElm)
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 += Number(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 =
Number(flexColumns[0].minWidth || 80) + totalFlexWidth
} else {
const allColumnsWidth = flexColumns.reduce(
(prev, column) => prev + Number(column.minWidth || 80),
0
)
const flexWidthPerPixel = totalFlexWidth / allColumnsWidth
let noneFirstWidth = 0
flexColumns.forEach((column, index) => {
if (index === 0) return
const flexWidth = Math.floor(
Number(column.minWidth || 80) * flexWidthPerPixel
)
noneFirstWidth += flexWidth
column.realWidth = Number(column.minWidth || 80) + flexWidth
})
flexColumns[0].realWidth =
Number(flexColumns[0].minWidth || 80) +
totalFlexWidth -
noneFirstWidth
}
} else {
// HAVE HORIZONTAL SCROLL BAR
this.scrollX.value = true
flexColumns.forEach(function (column) {
column.realWidth = Number(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 = Number(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 += Number(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 += Number(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