1
0
mirror of https://github.com/nolanlawson/pinafore synced 2025-01-27 12:09:18 +01:00

119 lines
3.8 KiB
JavaScript
Raw Normal View History

2018-01-25 08:23:14 -08:00
import { mark, stop } from '../../_utils/marks'
2018-01-30 21:55:23 -08:00
import { RealmStore } from '../../_utils/RealmStore'
import { reselect } from '../../_utils/reselect'
2018-03-22 22:31:29 -07:00
const RENDER_BUFFER_FACTOR = 2.5
2018-01-20 15:37:40 -08:00
2018-01-30 21:55:23 -08:00
class VirtualListStore extends RealmStore {
2018-02-08 22:29:29 -08:00
constructor (state) {
2018-01-30 21:55:23 -08:00
super(state, /* maxSize */ 10)
2018-01-17 19:41:37 -08:00
}
}
2018-01-31 08:37:59 -08:00
const virtualListStore = new VirtualListStore()
2018-02-08 18:54:48 -08:00
virtualListStore.computeForRealm('items', null)
2018-01-30 21:55:23 -08:00
virtualListStore.computeForRealm('showFooter', false)
2018-01-31 08:37:59 -08:00
virtualListStore.computeForRealm('footerHeight', 0)
2018-02-11 19:15:21 -08:00
virtualListStore.computeForRealm('showHeader', false)
virtualListStore.computeForRealm('headerHeight', 0)
2018-01-30 21:55:23 -08:00
virtualListStore.computeForRealm('scrollTop', 0)
virtualListStore.computeForRealm('scrollHeight', 0)
virtualListStore.computeForRealm('offsetHeight', 0)
virtualListStore.computeForRealm('listOffset', 0)
2018-01-31 08:37:59 -08:00
virtualListStore.computeForRealm('itemHeights', {})
2018-01-27 08:13:28 -08:00
virtualListStore.compute('rawVisibleItems',
['items', 'scrollTop', 'itemHeights', 'offsetHeight', 'showHeader', 'headerHeight', 'listOffset'],
(items, scrollTop, itemHeights, offsetHeight, showHeader, headerHeight, listOffset) => {
window.rawVisibleItemsComputed = (window.rawVisibleItemsComputed || 0) + 1
2018-02-08 22:29:29 -08:00
mark('compute visibleItems')
if (!items) {
return null
2018-01-15 17:25:32 -08:00
}
let effectiveScrollTop = scrollTop - listOffset
let renderBuffer = RENDER_BUFFER_FACTOR * offsetHeight
2018-02-08 22:29:29 -08:00
let visibleItems = []
2018-02-11 19:15:21 -08:00
let totalOffset = showHeader ? headerHeight : 0
2018-02-08 22:29:29 -08:00
let len = items.length
let i = -1
while (++i < len) {
let key = items[i]
let height = itemHeights[key] || 0
let currentOffset = totalOffset
totalOffset += height
let isAboveViewport = (currentOffset < effectiveScrollTop)
if (isAboveViewport) {
if ((effectiveScrollTop - height - renderBuffer) > currentOffset) {
continue // above the area we want to render
2018-02-08 22:29:29 -08:00
}
} else {
if (currentOffset > (effectiveScrollTop + offsetHeight + renderBuffer)) {
break // below the area we want to render
2018-02-08 22:29:29 -08:00
}
}
visibleItems.push({
offset: currentOffset,
key: key,
index: i
})
2018-01-15 17:25:32 -08:00
}
2018-02-08 22:29:29 -08:00
stop('compute visibleItems')
return visibleItems
2018-01-15 17:25:32 -08:00
})
reselect(virtualListStore, 'visibleItems', 'rawVisibleItems')
2018-01-21 16:07:11 -08:00
virtualListStore.compute('heightWithoutFooter',
2018-02-11 19:15:21 -08:00
['items', 'itemHeights', 'showHeader', 'headerHeight'],
(items, itemHeights, showHeader, headerHeight) => {
2018-02-08 22:29:29 -08:00
if (!items) {
return 0
}
2018-02-11 19:15:21 -08:00
let sum = showHeader ? headerHeight : 0
2018-02-08 22:29:29 -08:00
let i = -1
let len = items.length
while (++i < len) {
sum += itemHeights[items[i]] || 0
}
return sum
})
2018-01-21 16:07:11 -08:00
virtualListStore.compute('height',
['heightWithoutFooter', 'showFooter', 'footerHeight'],
(heightWithoutFooter, showFooter, footerHeight) => {
2018-02-08 22:29:29 -08:00
return showFooter ? (heightWithoutFooter + footerHeight) : heightWithoutFooter
})
2018-01-21 16:07:11 -08:00
2018-02-04 10:05:01 -08:00
virtualListStore.compute('length', ['items'], (items) => items ? items.length : 0)
2018-01-17 23:00:33 -08:00
2018-01-24 18:04:25 -08:00
virtualListStore.compute('allVisibleItemsHaveHeight',
['visibleItems', 'itemHeights'],
(visibleItems, itemHeights) => {
2018-02-08 22:29:29 -08:00
if (!visibleItems) {
return false
}
for (let visibleItem of visibleItems) {
if (!itemHeights[visibleItem.key]) {
return false
}
}
return true
})
2018-01-24 18:04:25 -08:00
if (process.browser && process.env.NODE_ENV !== 'production') {
window.virtualListStore = virtualListStore
virtualListStore.observe('visibleItems', () => {
window.visibleItemsChangedCount = (window.visibleItemsChangedCount || 0) + 1
})
virtualListStore.observe('rawVisibleItems', () => {
window.rawVisibleItemsChangedCount = (window.rawVisibleItemsChangedCount || 0) + 1
})
}
export {
virtualListStore
2018-02-08 22:29:29 -08:00
}