UNPKG

@ionic/core

Version:
234 lines (227 loc) • 12.5 kB
/*! * (C) Ionic http://ionicframework.com - MIT License */ import { r as registerInstance, c as createEvent, w as writeTask, d as readTask, h, f as getElement, e as Host } from './index-527b9e34.js'; import { f as findClosestIonContent, p as printIonContentErrorMsg, g as getScrollElement } from './index-e919e353.js'; import { b as getIonMode, c as config } from './ionic-global-ca86cf32.js'; import { E as ENABLE_HTML_CONTENT_DEFAULT, a as sanitizeDOMString } from './config-49c88215.js'; import './helpers-78efeec3.js'; import './index-738d7504.js'; const infiniteScrollCss = "ion-infinite-scroll{display:none;width:100%}.infinite-scroll-enabled{display:block}"; const IonInfiniteScrollStyle0 = infiniteScrollCss; const InfiniteScroll = class { constructor(hostRef) { registerInstance(this, hostRef); this.ionInfinite = createEvent(this, "ionInfinite", 7); this.thrPx = 0; this.thrPc = 0; /** * didFire exists so that ionInfinite * does not fire multiple times if * users continue to scroll after * scrolling into the infinite * scroll threshold. */ this.didFire = false; this.isBusy = false; this.onScroll = () => { const scrollEl = this.scrollEl; if (!scrollEl || !this.canStart()) { return 1; } const infiniteHeight = this.el.offsetHeight; if (infiniteHeight === 0) { // if there is no height of this element then do nothing return 2; } const scrollTop = scrollEl.scrollTop; const scrollHeight = scrollEl.scrollHeight; const height = scrollEl.offsetHeight; const threshold = this.thrPc !== 0 ? height * this.thrPc : this.thrPx; const distanceFromInfinite = this.position === 'bottom' ? scrollHeight - infiniteHeight - scrollTop - threshold - height : scrollTop - infiniteHeight - threshold; if (distanceFromInfinite < 0) { if (!this.didFire) { this.isLoading = true; this.didFire = true; this.ionInfinite.emit(); return 3; } } return 4; }; this.isLoading = false; this.threshold = '15%'; this.disabled = false; this.position = 'bottom'; } thresholdChanged() { const val = this.threshold; if (val.lastIndexOf('%') > -1) { this.thrPx = 0; this.thrPc = parseFloat(val) / 100; } else { this.thrPx = parseFloat(val); this.thrPc = 0; } } disabledChanged() { const disabled = this.disabled; if (disabled) { this.isLoading = false; this.isBusy = false; } this.enableScrollEvents(!disabled); } async connectedCallback() { const contentEl = findClosestIonContent(this.el); if (!contentEl) { printIonContentErrorMsg(this.el); return; } this.scrollEl = await getScrollElement(contentEl); this.thresholdChanged(); this.disabledChanged(); if (this.position === 'top') { writeTask(() => { if (this.scrollEl) { this.scrollEl.scrollTop = this.scrollEl.scrollHeight - this.scrollEl.clientHeight; } }); } } disconnectedCallback() { this.enableScrollEvents(false); this.scrollEl = undefined; } /** * Call `complete()` within the `ionInfinite` output event handler when * your async operation has completed. For example, the `loading` * state is while the app is performing an asynchronous operation, * such as receiving more data from an AJAX request to add more items * to a data list. Once the data has been received and UI updated, you * then call this method to signify that the loading has completed. * This method will change the infinite scroll's state from `loading` * to `enabled`. */ async complete() { const scrollEl = this.scrollEl; if (!this.isLoading || !scrollEl) { return; } this.isLoading = false; if (this.position === 'top') { /** * New content is being added at the top, but the scrollTop position stays the same, * which causes a scroll jump visually. This algorithm makes sure to prevent this. * (Frame 1) * - complete() is called, but the UI hasn't had time to update yet. * - Save the current content dimensions. * - Wait for the next frame using _dom.read, so the UI will be updated. * (Frame 2) * - Read the new content dimensions. * - Calculate the height difference and the new scroll position. * - Delay the scroll position change until other possible dom reads are done using _dom.write to be performant. * (Still frame 2, if I'm correct) * - Change the scroll position (= visually maintain the scroll position). * - Change the state to re-enable the InfiniteScroll. * - This should be after changing the scroll position, or it could * cause the InfiniteScroll to be triggered again immediately. * (Frame 3) * Done. */ this.isBusy = true; // ******** DOM READ **************** // Save the current content dimensions before the UI updates const prev = scrollEl.scrollHeight - scrollEl.scrollTop; // ******** DOM READ **************** requestAnimationFrame(() => { readTask(() => { // UI has updated, save the new content dimensions const scrollHeight = scrollEl.scrollHeight; // New content was added on top, so the scroll position should be changed immediately to prevent it from jumping around const newScrollTop = scrollHeight - prev; // ******** DOM WRITE **************** requestAnimationFrame(() => { writeTask(() => { scrollEl.scrollTop = newScrollTop; this.isBusy = false; this.didFire = false; }); }); }); }); } else { this.didFire = false; } } canStart() { return !this.disabled && !this.isBusy && !!this.scrollEl && !this.isLoading; } enableScrollEvents(shouldListen) { if (this.scrollEl) { if (shouldListen) { this.scrollEl.addEventListener('scroll', this.onScroll); } else { this.scrollEl.removeEventListener('scroll', this.onScroll); } } } render() { const mode = getIonMode(this); const disabled = this.disabled; return (h(Host, { key: 'e844956795f69be33396ce4480aa7a54ad01b28c', class: { [mode]: true, 'infinite-scroll-loading': this.isLoading, 'infinite-scroll-enabled': !disabled, } })); } get el() { return getElement(this); } static get watchers() { return { "threshold": ["thresholdChanged"], "disabled": ["disabledChanged"] }; } }; InfiniteScroll.style = IonInfiniteScrollStyle0; const infiniteScrollContentIosCss = "ion-infinite-scroll-content{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;-ms-flex-pack:center;justify-content:center;min-height:84px;text-align:center;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.infinite-loading{margin-left:0;margin-right:0;margin-top:0;margin-bottom:32px;display:none;width:100%}.infinite-loading-text{-webkit-margin-start:32px;margin-inline-start:32px;-webkit-margin-end:32px;margin-inline-end:32px;margin-top:4px;margin-bottom:0}.infinite-scroll-loading ion-infinite-scroll-content>.infinite-loading{display:block}.infinite-scroll-content-ios .infinite-loading-text{color:var(--ion-color-step-600, var(--ion-text-color-step-400, #666666))}.infinite-scroll-content-ios .infinite-loading-spinner .spinner-lines-ios line,.infinite-scroll-content-ios .infinite-loading-spinner .spinner-lines-small-ios line,.infinite-scroll-content-ios .infinite-loading-spinner .spinner-crescent circle{stroke:var(--ion-color-step-600, var(--ion-text-color-step-400, #666666))}.infinite-scroll-content-ios .infinite-loading-spinner .spinner-bubbles circle,.infinite-scroll-content-ios .infinite-loading-spinner .spinner-circles circle,.infinite-scroll-content-ios .infinite-loading-spinner .spinner-dots circle{fill:var(--ion-color-step-600, var(--ion-text-color-step-400, #666666))}"; const IonInfiniteScrollContentIosStyle0 = infiniteScrollContentIosCss; const infiniteScrollContentMdCss = "ion-infinite-scroll-content{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;-ms-flex-pack:center;justify-content:center;min-height:84px;text-align:center;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.infinite-loading{margin-left:0;margin-right:0;margin-top:0;margin-bottom:32px;display:none;width:100%}.infinite-loading-text{-webkit-margin-start:32px;margin-inline-start:32px;-webkit-margin-end:32px;margin-inline-end:32px;margin-top:4px;margin-bottom:0}.infinite-scroll-loading ion-infinite-scroll-content>.infinite-loading{display:block}.infinite-scroll-content-md .infinite-loading-text{color:var(--ion-color-step-600, var(--ion-text-color-step-400, #666666))}.infinite-scroll-content-md .infinite-loading-spinner .spinner-lines-md line,.infinite-scroll-content-md .infinite-loading-spinner .spinner-lines-small-md line,.infinite-scroll-content-md .infinite-loading-spinner .spinner-crescent circle{stroke:var(--ion-color-step-600, var(--ion-text-color-step-400, #666666))}.infinite-scroll-content-md .infinite-loading-spinner .spinner-bubbles circle,.infinite-scroll-content-md .infinite-loading-spinner .spinner-circles circle,.infinite-scroll-content-md .infinite-loading-spinner .spinner-dots circle{fill:var(--ion-color-step-600, var(--ion-text-color-step-400, #666666))}"; const IonInfiniteScrollContentMdStyle0 = infiniteScrollContentMdCss; const InfiniteScrollContent = class { constructor(hostRef) { registerInstance(this, hostRef); this.customHTMLEnabled = config.get('innerHTMLTemplatesEnabled', ENABLE_HTML_CONTENT_DEFAULT); this.loadingSpinner = undefined; this.loadingText = undefined; } componentDidLoad() { if (this.loadingSpinner === undefined) { const mode = getIonMode(this); this.loadingSpinner = config.get('infiniteLoadingSpinner', config.get('spinner', mode === 'ios' ? 'lines' : 'crescent')); } } renderLoadingText() { const { customHTMLEnabled, loadingText } = this; if (customHTMLEnabled) { return h("div", { class: "infinite-loading-text", innerHTML: sanitizeDOMString(loadingText) }); } return h("div", { class: "infinite-loading-text" }, this.loadingText); } render() { const mode = getIonMode(this); return (h(Host, { key: '7c16060dcfe2a0b0fb3e2f8f4c449589a76f1baa', class: { [mode]: true, // Used internally for styling [`infinite-scroll-content-${mode}`]: true, } }, h("div", { key: 'a94f4d8746e053dc718f97520bd7e48cb316443a', class: "infinite-loading" }, this.loadingSpinner && (h("div", { key: '10143d5d2a50a2a2bc5de1cee8e7ab51263bcf23', class: "infinite-loading-spinner" }, h("ion-spinner", { key: '8846e88191690d9c61a0b462889ed56fbfed8b0d', name: this.loadingSpinner }))), this.loadingText !== undefined && this.renderLoadingText()))); } }; InfiniteScrollContent.style = { ios: IonInfiniteScrollContentIosStyle0, md: IonInfiniteScrollContentMdStyle0 }; export { InfiniteScroll as ion_infinite_scroll, InfiniteScrollContent as ion_infinite_scroll_content };