element-plus
Version:
A Component Library for Vue3.0
502 lines (431 loc) • 14.1 kB
text/typescript
import {
computed,
defineComponent,
getCurrentInstance,
ref,
nextTick,
onMounted,
onUpdated,
resolveDynamicComponent,
h,
} from 'vue'
import { hasOwn } from '@vue/shared'
import memo from 'lodash/memoize'
import { isNumber, isString, $ } from '@element-plus/utils/util'
import isServer from '@element-plus/utils/isServer'
import useWheel from '../hooks/useWheel'
import Scrollbar from '../components/scrollbar'
import { getScrollDir, isHorizontal, getRTLOffsetType } from '../utils'
import {
DefaultListProps,
AUTO_ALIGNMENT,
BACKWARD,
FORWARD,
RTL,
HORIZONTAL,
ITEM_RENDER_EVT,
SCROLL_EVT,
RTL_OFFSET_NAG,
RTL_OFFSET_POS_DESC,
} from '../defaults'
import type { VNode, CSSProperties } from 'vue'
import type { ListConstructorProps, Alignment } from '../types'
const createList = ({
name,
getOffset,
getItemSize,
getItemOffset,
getEstimatedTotalSize,
getStartIndexForOffset,
getStopIndexForStartIndex,
initCache,
clearCache,
validateProps,
}: ListConstructorProps<typeof DefaultListProps>) => {
return defineComponent({
name: name ?? 'ElVirtualList',
props: DefaultListProps,
emits: [ITEM_RENDER_EVT, SCROLL_EVT],
setup(props, { emit, expose }) {
validateProps(props)
const instance = getCurrentInstance()
const dynamicSizeCache = ref(initCache(props, instance))
// refs
// here windowRef and innerRef can be type of HTMLElement
// or user defined component type, depends on the type passed
// by user
const windowRef = ref<HTMLElement>(null)
const innerRef = ref<HTMLElement>(null)
const scrollbarRef = ref(null)
const states = ref({
isScrolling: false,
scrollDir: 'forward',
scrollOffset: isNumber(props.initScrollOffset) ? props.initScrollOffset : 0,
updateRequested: false,
isScrollbarDragging: false,
})
// computed
const itemsToRender = computed(() => {
const { total, cache } = props
const { isScrolling, scrollDir, scrollOffset } = $(states)
if (total === 0) {
return [0, 0, 0, 0]
}
const startIndex = getStartIndexForOffset(
props,
scrollOffset,
$(dynamicSizeCache),
)
const stopIndex = getStopIndexForStartIndex(
props,
startIndex,
scrollOffset,
$(dynamicSizeCache),
)
const cacheBackward =
!isScrolling || scrollDir === BACKWARD
? Math.max(1, cache)
: 1
const cacheForward =
!isScrolling || scrollDir === FORWARD
? Math.max(1, cache)
: 1
return [
Math.max(0, startIndex - cacheBackward),
Math.max(0, Math.min(total - 1, stopIndex + cacheForward)),
startIndex,
stopIndex,
]
})
const estimatedTotalSize = computed(() => getEstimatedTotalSize(props, $(dynamicSizeCache)))
const _isHorizontal = computed(() => isHorizontal(props.layout))
const windowStyle = computed(() => ([
{
position: 'relative',
overflow: 'hidden',
WebkitOverflowScrolling: 'touch',
willChange: 'transform',
}, {
direction: props.direction,
height: isNumber(props.height) ? `${props.height}px` : props.height,
width: isNumber(props.width) ? `${props.width}px` : props.width,
...props.style,
},
]))
const innerStyle = computed(() => {
const size = $(estimatedTotalSize)
const horizontal = $(_isHorizontal)
return {
height: horizontal ? '100%' : `${size}px`,
pointerEvents: $(states).isScrolling ? 'none' : undefined,
width: horizontal ? `${size}px` : '100%',
}
})
const clientSize = computed(() => _isHorizontal.value ? props.width : props.height)
// methods
const {
onWheel,
} = useWheel({
atStartEdge: computed(() => states.value.scrollOffset <= 0),
atEndEdge: computed(() => states.value.scrollOffset >= estimatedTotalSize.value),
layout: computed(() => props.layout),
}, offset => {
scrollbarRef.value.onMouseUp?.()
scrollTo(
Math.min(
states.value.scrollOffset + offset,
estimatedTotalSize.value - (clientSize.value as number),
),
)
})
const emitEvents = () => {
const { total } = props
if (total > 0) {
const [cacheStart, cacheEnd, visibleStart, visibleEnd] = $(itemsToRender)
emit(ITEM_RENDER_EVT, cacheStart, cacheEnd, visibleStart, visibleEnd)
}
const { scrollDir, scrollOffset, updateRequested } = $(states)
emit(SCROLL_EVT, scrollDir, scrollOffset, updateRequested)
}
const scrollVertically = (e: Event) => {
const { clientHeight, scrollHeight, scrollTop } = e.currentTarget as HTMLElement
const _states = $(states)
if (_states.scrollOffset === scrollTop) {
return
}
const scrollOffset = Math.max(
0,
Math.min(scrollTop, scrollHeight - clientHeight),
)
states.value = {
..._states,
isScrolling: true,
scrollDir: getScrollDir(_states.scrollOffset, scrollOffset),
scrollOffset,
updateRequested: false,
}
nextTick(resetIsScrolling)
}
const scrollHorizontally = (e: Event) => {
const { clientWidth, scrollLeft, scrollWidth } = (e.currentTarget) as HTMLElement
const _states = $(states)
if (_states.scrollOffset === scrollLeft) {
return
}
const { direction } = props
let scrollOffset = scrollLeft
if (direction === RTL) {
// TRICKY According to the spec, scrollLeft should be negative for RTL aligned elements.
// This is not the case for all browsers though (e.g. Chrome reports values as positive, measured relative to the left).
// It's also easier for this component if we convert offsets to the same format as they would be in for ltr.
// So the simplest solution is to determine which browser behavior we're dealing with, and convert based on it.
switch (getRTLOffsetType()) {
case RTL_OFFSET_NAG: {
scrollOffset = -scrollLeft
break
}
case RTL_OFFSET_POS_DESC: {
scrollOffset = scrollWidth - clientWidth - scrollLeft
break
}
}
}
scrollOffset = Math.max(
0,
Math.min(scrollOffset, scrollWidth - clientWidth),
)
states.value = {
..._states,
isScrolling: true,
scrollDir: getScrollDir(_states.scrollOffset, scrollOffset),
scrollOffset,
updateRequested: false,
}
nextTick(resetIsScrolling)
}
const onScroll = (e: Event) => {
$(_isHorizontal) ? scrollHorizontally(e) : scrollVertically(e)
emitEvents()
}
const onScrollbarScroll = (distanceToGo: number, totalSteps: number) => {
const offset = (estimatedTotalSize.value - (clientSize.value as number)) / totalSteps * distanceToGo
scrollTo(
Math.min(
estimatedTotalSize.value - (clientSize.value as number),
offset,
),
)
}
// eslint-disable-next-line @typescript-eslint/no-unused-vars
const getItemStyleCache = memo((_: any, __: any, ___: any) => ({}))
const scrollTo = (offset: number) => {
offset = Math.max(offset, 0)
if (offset === $(states).scrollOffset) {
return
}
states.value = {
...$(states),
scrollOffset: offset,
scrollDir: getScrollDir($(states).scrollOffset, offset),
updateRequested: true,
}
nextTick(resetIsScrolling)
}
const scrollToItem = (idx: number, alignment: Alignment = AUTO_ALIGNMENT) => {
const { scrollOffset } = $(states)
idx = Math.max(0, Math.min(idx, props.total - 1))
scrollTo(
getOffset(
props,
idx,
alignment,
scrollOffset,
$(dynamicSizeCache),
),
)
}
const getItemStyle = (idx: number) => {
const { direction, itemSize, layout } = props
const itemStyleCache = getItemStyleCache(
clearCache && itemSize,
clearCache && layout,
clearCache && direction,
)
let style: CSSProperties
if (hasOwn(itemStyleCache, String(idx))) {
style = itemStyleCache[idx]
} else {
const offset = getItemOffset(props, idx, $(dynamicSizeCache))
const size = getItemSize(props, idx, $(dynamicSizeCache))
const horizontal = $(_isHorizontal)
const isRtl = direction === RTL
const offsetHorizontal = horizontal ? offset : 0
itemStyleCache[idx] = style = {
position: 'absolute',
left: isRtl ? undefined : `${offsetHorizontal}px`,
right: isRtl ? `${offsetHorizontal}px` : undefined,
top: !horizontal ? `${offset}px` : 0,
height: !horizontal ? `${size}px` : '100%',
width: horizontal ? `${size}px` : '100%',
}
}
return style
}
// TODO:
// perf optimization here, reset isScrolling with debounce.
const resetIsScrolling = () => {
// timer = null
states.value.isScrolling = false
nextTick(() => {
getItemStyleCache(-1, null, null)
})
}
// life cycles
onMounted(() => {
if (isServer) return
const { initScrollOffset } = props
const windowElement = $(windowRef)
if (isNumber(initScrollOffset) && windowElement !== null) {
if ($(_isHorizontal)) {
windowElement.scrollLeft = initScrollOffset
} else {
windowElement.scrollTop = initScrollOffset
}
}
emitEvents()
})
onUpdated(() => {
const { direction, layout } = props
const { scrollOffset, updateRequested } = $(states)
if (updateRequested && $(windowRef) !== null) {
const windowElement = $(windowRef)
if (layout === HORIZONTAL) {
if (direction === RTL) {
// TRICKY According to the spec, scrollLeft should be negative for RTL aligned elements.
// This is not the case for all browsers though (e.g. Chrome reports values as positive, measured relative to the left).
// So we need to determine which browser behavior we're dealing with, and mimic it.
switch (getRTLOffsetType()) {
case 'negative': {
windowElement.scrollLeft = -scrollOffset
break
}
case 'positive-ascending': {
windowElement.scrollLeft = scrollOffset
break
}
default: {
const { clientWidth, scrollWidth } = windowElement
windowElement.scrollLeft = scrollWidth - clientWidth - scrollOffset
break
}
}
} else {
windowElement.scrollLeft = scrollOffset
}
} else {
windowElement.scrollTop = scrollOffset
}
}
})
const api = {
clientSize,
estimatedTotalSize,
windowStyle,
windowRef,
innerRef,
innerStyle,
itemsToRender,
scrollbarRef,
states,
getItemStyle,
onScroll,
onScrollbarScroll,
onWheel,
scrollTo,
scrollToItem,
}
expose({
windowRef,
innerRef,
getItemStyleCache,
scrollTo,
scrollToItem,
states,
})
return api
},
render(ctx: any) {
const {
$slots,
className,
clientSize,
containerElement,
data,
getItemStyle,
innerElement,
itemsToRender,
innerStyle,
layout,
total,
onScroll,
onScrollbarScroll,
onWheel,
states,
useIsScrolling,
windowStyle,
} = ctx
const [start, end] = itemsToRender
const Container = resolveDynamicComponent(containerElement)
const Inner = resolveDynamicComponent(innerElement)
const children = []
if (total > 0) {
for (let i = start; i <= end; i++) {
children.push(
$slots.default?.({
data,
key: i,
index: i,
isScrolling: useIsScrolling ? states.isScrolling : undefined,
style: getItemStyle(i),
}),
)
}
}
const InnerNode = [h(Inner as VNode, {
style: innerStyle,
ref: 'innerRef',
}, !isString(Inner) ? {
default: () => children,
} : children)]
const scrollbar = h(Scrollbar, {
ref: 'scrollbarRef',
clientSize,
layout,
onScroll: onScrollbarScroll,
ratio: (clientSize * 100) / this.estimatedTotalSize,
scrollFrom: states.scrollOffset / (this.estimatedTotalSize - clientSize),
total,
visible: true,
})
const listContainer = h(Container as VNode, {
class: className,
style: windowStyle,
onScroll,
onWheel,
ref: 'windowRef',
key: 0,
}, !isString(Container)
? { default: () => [InnerNode] }
: [InnerNode],
)
return h('div', {
key: 0,
class: 'el-vl__wrapper',
}, [
listContainer,
scrollbar,
])
},
})
}
export default createList