forked from cybrespace/pinafore
70 lines
1.8 KiB
HTML
70 lines
1.8 KiB
HTML
|
<div class="virtual-list" ref:node style="height: {{height}}px;">
|
||
|
{{#each visibleItems as visibleItem, index}}
|
||
|
<VirtualListItem :component
|
||
|
:intersectionObserver
|
||
|
virtualOffset="{{visibleItem.offset}}"
|
||
|
virtualProps="{{visibleItem.props}}"
|
||
|
virtualIndex="{{index}}" />
|
||
|
{{/each}}
|
||
|
</div>
|
||
|
<style>
|
||
|
.virtual-list {
|
||
|
position: relative;
|
||
|
}
|
||
|
</style>
|
||
|
<script>
|
||
|
import VirtualListItem from './VirtualListItem'
|
||
|
|
||
|
function sum(arr) {
|
||
|
return arr.reduce((a, b) => a + b, 0)
|
||
|
}
|
||
|
|
||
|
export default {
|
||
|
oncreate() {
|
||
|
let intersectionObserver = new IntersectionObserver((entries) => {
|
||
|
let totalHeight = sum(entries.map(entry => entry.boundingClientRect.height))
|
||
|
let offset = 0
|
||
|
let offsets = []
|
||
|
entries.forEach(entry => {
|
||
|
offsets.push(offset)
|
||
|
offset += entry.boundingClientRect.height
|
||
|
})
|
||
|
this.set({
|
||
|
height: totalHeight,
|
||
|
offsets: offsets
|
||
|
})
|
||
|
console.log('entries', entries.map(entry => entry.target.getAttribute('data-virtual-index')))
|
||
|
}, {
|
||
|
root: this.refs.node
|
||
|
})
|
||
|
this.set({
|
||
|
intersectionObserver: intersectionObserver
|
||
|
})
|
||
|
},
|
||
|
ondestroy() {
|
||
|
let intersectionObserver = this.get('intersectionObserver')
|
||
|
if (intersectionObserver) {
|
||
|
intersectionObserver.disconnect()
|
||
|
}
|
||
|
},
|
||
|
computed: {
|
||
|
visibleItems: (items, offsets) => {
|
||
|
return items.map((item, idx) => ({
|
||
|
props: item,
|
||
|
offset: offsets[idx]
|
||
|
}))
|
||
|
}
|
||
|
},
|
||
|
data: () => ({
|
||
|
scrollHeight: 0,
|
||
|
component: null,
|
||
|
intersectionObserver: null,
|
||
|
items: [],
|
||
|
offsets: [],
|
||
|
height: 400
|
||
|
}),
|
||
|
components: {
|
||
|
VirtualListItem
|
||
|
}
|
||
|
}
|
||
|
</script>
|