|
| 1 | +import Vue from 'vue'; |
| 2 | +import { throttle } from './util'; |
| 3 | + |
| 4 | +let delta = { |
| 5 | + total: 0, |
| 6 | + direct: '', |
| 7 | + last_top: 0, |
| 8 | + page_type: '', |
| 9 | + start_index: 0, |
| 10 | + all_padding: 0, |
| 11 | + padding_top: 0, |
| 12 | + bench_padding: 0 |
| 13 | +}; |
| 14 | + |
| 15 | +Vue.component('virtual-list', { |
| 16 | + |
| 17 | + props: { |
| 18 | + unit: { |
| 19 | + type: Number, |
| 20 | + required: true |
| 21 | + }, |
| 22 | + remain: { |
| 23 | + type: Number, |
| 24 | + required: true |
| 25 | + }, |
| 26 | + pageCounts: { |
| 27 | + type: Number, |
| 28 | + required: true |
| 29 | + } |
| 30 | + }, |
| 31 | + |
| 32 | + methods: { |
| 33 | + onScroll: throttle(function () { |
| 34 | + let cont = this.$refs.container; |
| 35 | + let listbox = this.$refs.listbox; |
| 36 | + |
| 37 | + let scrollTop = cont.scrollTop; |
| 38 | + let viewHeight = cont.offsetHeight; |
| 39 | + let listHeight = listbox.offsetHeight; |
| 40 | + |
| 41 | + this.recordDirect(scrollTop); |
| 42 | + |
| 43 | + // scroll to top |
| 44 | + if (scrollTop === 0) { |
| 45 | + this.$emit('top'); |
| 46 | + } |
| 47 | + |
| 48 | + // scroll to bottom |
| 49 | + let paddingBottom = delta.all_padding - delta.padding_top; |
| 50 | + if (listHeight <= scrollTop + viewHeight + paddingBottom) { |
| 51 | + this.showNext(); |
| 52 | + } |
| 53 | + |
| 54 | + if (delta.direct === 'UP' && scrollTop < delta.padding_top) { |
| 55 | + this.showPrev(); |
| 56 | + } |
| 57 | + }, 16, true, true), |
| 58 | + |
| 59 | + recordDirect (scrollTop) { |
| 60 | + if (!delta.last_top) { |
| 61 | + delta.last_top = scrollTop; |
| 62 | + } else { |
| 63 | + delta.direct = delta.last_top > scrollTop ? 'UP' : 'DOWN'; |
| 64 | + delta.last_top = scrollTop; |
| 65 | + } |
| 66 | + }, |
| 67 | + |
| 68 | + showNext () { |
| 69 | + delta.page_type = 'NEXT'; |
| 70 | + this.$emit('bottom'); |
| 71 | + }, |
| 72 | + |
| 73 | + showPrev () { |
| 74 | + delta.page_type = 'PREV'; |
| 75 | + this.$forceUpdate(); |
| 76 | + this.$emit('prev'); |
| 77 | + }, |
| 78 | + |
| 79 | + filter (items) { |
| 80 | + let length = items.length; |
| 81 | + let nowStartIndex, udf, list = []; |
| 82 | + |
| 83 | + if (!delta.total) { |
| 84 | + delta.total = length; |
| 85 | + } |
| 86 | + |
| 87 | + if (delta.page_type === 'PREV') { |
| 88 | + // already the first page |
| 89 | + if (delta.start_index === 0) { |
| 90 | + list = items.slice(0, this.pageCounts); |
| 91 | + } else { |
| 92 | + list = items.filter((item, index) => { |
| 93 | + if (index === delta.start_index - this.pageCounts) { |
| 94 | + nowStartIndex = index; |
| 95 | + } |
| 96 | + |
| 97 | + return index >= (delta.start_index - this.pageCounts) |
| 98 | + && index <= (delta.start_index - 1); |
| 99 | + }); |
| 100 | + |
| 101 | + if (nowStartIndex !== udf) { |
| 102 | + delta.start_index = nowStartIndex; |
| 103 | + } |
| 104 | + |
| 105 | + delta.padding_top = delta.start_index * this.unit; |
| 106 | + } |
| 107 | + } else { |
| 108 | + // virtual list has no any increase |
| 109 | + if (length === delta.total) { |
| 110 | + list = items; |
| 111 | + } else { |
| 112 | + list = items.filter((item, index) => { |
| 113 | + if (index === delta.start_index + this.pageCounts) { |
| 114 | + nowStartIndex = index; |
| 115 | + } |
| 116 | + |
| 117 | + return index >= (delta.start_index + this.pageCounts) |
| 118 | + && index < (delta.start_index + this.pageCounts * 2); |
| 119 | + }); |
| 120 | + |
| 121 | + if (nowStartIndex !== udf) { |
| 122 | + delta.start_index = nowStartIndex; |
| 123 | + } |
| 124 | + |
| 125 | + // item counts of all virtual list |
| 126 | + delta.total = length; |
| 127 | + // all padding pixel, except remain in viewport items |
| 128 | + delta.all_padding = (length - this.remain) * this.unit; |
| 129 | + // padding-top piexl |
| 130 | + delta.padding_top = delta.start_index * this.unit; |
| 131 | + } |
| 132 | + } |
| 133 | + |
| 134 | + return list; |
| 135 | + } |
| 136 | + }, |
| 137 | + |
| 138 | + mounted () { |
| 139 | + delta.bench_padding = Math.ceil(this.remain / 2) * this.unit; |
| 140 | + }, |
| 141 | + |
| 142 | + updated () { |
| 143 | + window.requestAnimationFrame(() => { |
| 144 | + this.$refs.container.scrollTop = delta.padding_top + delta.bench_padding; |
| 145 | + }); |
| 146 | + }, |
| 147 | + |
| 148 | + render (createElement) { |
| 149 | + let slots = this.$slots.default; |
| 150 | + let showList = this.filter(slots); |
| 151 | + |
| 152 | + return createElement('div', { |
| 153 | + 'ref': 'container', |
| 154 | + 'class': 'virtual-list', |
| 155 | + 'style': { |
| 156 | + 'overflow-y': 'auto', |
| 157 | + 'height': (this.remain * this.unit) + 'px' |
| 158 | + }, |
| 159 | + 'on': { |
| 160 | + 'scroll': this.onScroll |
| 161 | + } |
| 162 | + }, [ |
| 163 | + createElement('div', { |
| 164 | + 'ref': 'listbox', |
| 165 | + 'class': 'virtual-list-box', |
| 166 | + }, [ |
| 167 | + createElement('div', { |
| 168 | + 'class': 'virtual-list-box-padding', |
| 169 | + 'style': { |
| 170 | + 'padding-top': delta.padding_top + 'px', |
| 171 | + 'padding-bottom': (delta.all_padding - delta.padding_top) + 'px' |
| 172 | + } |
| 173 | + }, showList) |
| 174 | + ]) |
| 175 | + ]); |
| 176 | + } |
| 177 | +}); |
0 commit comments