import Vue from 'vue'; function getScrollContainer(el: Element | null): Element | null { if (el == null || el.tagName === 'BODY') return null; const style = window.getComputedStyle(el); if (style.getPropertyValue('overflow') === 'auto') { return el; } else { return getScrollContainer(el.parentElement); } } function getScrollPosition(el: Element | null): number { const container = getScrollContainer(el); return container == null ? window.scrollY : container.scrollTop; } function onScrollTop(el, cb) { const container = getScrollContainer(el) || window; const onScroll = ev => { if (!document.body.contains(el)) return; const pos = getScrollPosition(el); if (pos === 0) { cb(); container.removeEventListener('scroll', onscroll); } }; container.addEventListener('scroll', onScroll, { passive: true }); } export default (opts) => ({ data() { return { items: [], queue: [], offset: 0, fetching: true, moreFetching: false, inited: false, more: false, backed: false, isBackTop: false, }; }, computed: { empty(): boolean { return this.items.length == 0 && !this.fetching && this.inited; }, error(): boolean { return !this.fetching && !this.inited; }, }, watch: { pagination() { this.init(); } }, created() { opts.displayLimit = opts.displayLimit || 30; this.init(); this.$on('hook:activated', () => { this.isBackTop = false; }); this.$on('hook:deactivated', () => { this.isBackTop = window.scrollY === 0; }); }, methods: { updateItem(i, item) { Vue.set((this as any).items, i, item); }, reload() { this.items = []; this.init(); }, async init() { this.fetching = true; if (opts.before) opts.before(this); let params = typeof this.pagination.params === 'function' ? this.pagination.params(true) : this.pagination.params; if (params && params.then) params = await params; const endpoint = typeof this.pagination.endpoint === 'function' ? this.pagination.endpoint() : this.pagination.endpoint; await this.$root.api(endpoint, { limit: this.pagination.noPaging ? (this.pagination.limit || 10) : (this.pagination.limit || 10) + 1, ...params }).then(x => { if (!this.pagination.noPaging && (x.length === (this.pagination.limit || 10) + 1)) { x.pop(); this.items = x; this.more = true; } else { this.items = x; this.more = false; } this.offset = x.length; this.inited = true; this.fetching = false; if (opts.after) opts.after(this, null); }, e => { this.fetching = false; if (opts.after) opts.after(this, e); }); }, async fetchMore() { if (!this.more || this.moreFetching || this.items.length === 0) return; this.moreFetching = true; this.backed = true; let params = typeof this.pagination.params === 'function' ? this.pagination.params(false) : this.pagination.params; if (params && params.then) params = await params; const endpoint = typeof this.pagination.endpoint === 'function' ? this.pagination.endpoint() : this.pagination.endpoint; await this.$root.api(endpoint, { limit: (this.pagination.limit || 10) + 1, ...(this.pagination.offsetMode ? { offset: this.offset, } : { untilId: this.items[this.items.length - 1].id, }), ...params }).then(x => { if (x.length === (this.pagination.limit || 10) + 1) { x.pop(); this.items = this.items.concat(x); this.more = true; } else { this.items = this.items.concat(x); this.more = false; } this.offset += x.length; this.moreFetching = false; }, e => { this.moreFetching = false; }); }, prepend(item) { const isTop = this.isBackTop || (document.body.contains(this.$el) && (getScrollPosition(this.$el) === 0)); if (isTop) { // Prepend the item this.items.unshift(item); // オーバーフローしたら古いアイテムは捨てる if (this.items.length >= opts.displayLimit) { this.items = this.items.slice(0, opts.displayLimit); this.more = true; } } else { this.queue.push(item); onScrollTop(this.$el, () => { for (const item of this.queue) { this.prepend(item); } this.queue = []; }); } }, append(item) { this.items.push(item); }, remove(find) { this.items = this.items.filter(x => !find(x)); }, } });