<h1 class="sr-only">{label}</h1> <div class="timeline" role="feed" on:focusWithCapture="saveFocus(event)" on:blurWithCapture="clearFocus(event)" > {#await componentsPromise} <!-- awaiting promise --> {:then result} <svelte:component this={result.listComponent} component={result.listItemComponent} realm="{$currentInstance + '/' + timeline}" containerQuery=".container" {makeProps} items={$timelineItemIds} showFooter={$timelineInitialized && $runningUpdate} footerComponent={LoadingFooter} showHeader={$showHeader} headerComponent={MoreHeaderVirtualWrapper} {headerProps} {scrollToItem} on:scrollToBottom="onScrollToBottom()" on:scrollToTop="onScrollToTop()" on:scrollTopChanged="onScrollTopChanged(event)" on:initialized="initialize()" on:noNeedToScroll="onNoNeedToScroll()" /> {:catch error} <div>Error: component failed to load! Try reloading. {error}</div> {/await} </div> <script> import { store } from '../../_store/store' import Status from '../status/Status.html' import LoadingFooter from './LoadingFooter.html' import MoreHeaderVirtualWrapper from './MoreHeaderVirtualWrapper.html' import { importVirtualList, importList, importStatusVirtualListItem, importNotificationVirtualListItem } from '../../_utils/asyncModules' import { timelines } from '../../_static/timelines' import { getStatus as getStatusFromDatabase, getNotification as getNotificationFromDatabase } from '../../_database/timelines/getStatusOrNotification' import { fetchTimelineItemsOnScrollToBottom, setupTimeline, showMoreItemsForTimeline, showMoreItemsForThread, showMoreItemsForCurrentTimeline } from '../../_actions/timeline' import { focusWithCapture, blurWithCapture } from '../../_utils/events' import { scheduleIdleTask } from '../../_utils/scheduleIdleTask' import { mark, stop } from '../../_utils/marks' import isEqual from 'lodash-es/isEqual' import { doubleRAF } from '../../_utils/doubleRAF' import { observe } from 'svelte-extras' export default { oncreate () { console.log('timeline oncreate()') this.setupFocus() setupTimeline() this.restoreFocus() this.setupStreaming() }, ondestroy () { console.log('ondestroy') this.teardownFocus() }, data: () => ({ LoadingFooter, MoreHeaderVirtualWrapper, Status, scrollTop: 0 }), computed: { // For threads, it's simpler to just render all items as a pseudo-virtual list // due to need to scroll to the right item and thus calculate all item heights up-front. // Here we lazy-load both the virtual list component itself as well as the component // it renders. componentsPromise: ({ timelineType }) => { return Promise.all([ timelineType === 'status' ? importList() : importVirtualList(), timelineType === 'notifications' ? importNotificationVirtualListItem() : importStatusVirtualListItem() ]).then(results => ({ listComponent: results[0], listItemComponent: results[1] })) }, makeProps: ({ $currentInstance, timelineType, timelineValue }) => async (itemId) => { let res = { timelineType, timelineValue } if (timelineType === 'notifications') { res.notification = await getNotificationFromDatabase($currentInstance, itemId) } else { res.status = await getStatusFromDatabase($currentInstance, itemId) } return res }, label: ({ timeline, $currentInstance, timelineType, timelineValue }) => { if (timelines[timeline]) { return `${timelines[timeline].label} timeline for ${$currentInstance}` } switch (timelineType) { case 'tag': return `#${timelineValue} timeline for ${$currentInstance}` case 'status': return 'Status context' case 'account': return `Account #${timelineValue} on ${$currentInstance}` case 'list': return `List #${timelineValue} on ${$currentInstance}` case 'notifications': return `Notifications for ${$currentInstance}` } }, timelineType: ({ timeline }) => { return timeline.split('/')[0] }, timelineValue: ({ timeline }) => { return timeline.split('/').slice(-1)[0] }, // Scroll to the first item if this is a "status in own thread" timeline. // Don't scroll to the first item because it obscures the "back" button. scrollToItem: ({ timelineType, timelineValue, $firstTimelineItemId }) => ( timelineType === 'status' && $firstTimelineItemId && timelineValue !== $firstTimelineItemId && timelineValue ), itemIdsToAdd: ({ $itemIdsToAdd }) => $itemIdsToAdd, headerProps: ({ itemIdsToAdd }) => { return { count: itemIdsToAdd ? itemIdsToAdd.length : 0, onClick: showMoreItemsForCurrentTimeline } } }, store: () => store, events: { focusWithCapture, blurWithCapture }, methods: { observe, initialize () { let { initializeStarted } = this.get() if (initializeStarted) { return } this.set({initializeStarted: true}) mark('initializeTimeline') doubleRAF(() => { console.log('timeline initialized') this.store.set({timelineInitialized: true}) stop('initializeTimeline') }) }, onScrollTopChanged (scrollTop) { this.set({scrollTop: scrollTop}) }, onScrollToBottom () { let { timelineType } = this.get() let { timelineInitialized, runningUpdate } = this.store.get() if (!timelineInitialized || runningUpdate || timelineType === 'status') { // for status contexts, we've already fetched the whole thread return } let { currentInstance } = this.store.get() let { timeline } = this.get() fetchTimelineItemsOnScrollToBottom( currentInstance, timeline ) }, onScrollToTop () { let { shouldShowHeader } = this.store.get() if (shouldShowHeader) { this.store.setForCurrentTimeline({ showHeader: true, shouldShowHeader: false }) } }, setupStreaming () { let { currentInstance } = this.store.get() let { timeline, timelineType } = this.get() let handleItemIdsToAdd = () => { let { itemIdsToAdd } = this.get() if (!itemIdsToAdd || !itemIdsToAdd.length) { return } mark('handleItemIdsToAdd') let { scrollTop } = this.get() let { shouldShowHeader, showHeader } = this.store.get() if (timelineType === 'status') { // this is a thread, just insert the statuses already showMoreItemsForThread(currentInstance, timeline) } else if (scrollTop === 0 && !shouldShowHeader && !showHeader) { // if the user is scrolled to the top and we're not showing the header, then // just insert the statuses. this is "chat room mode" showMoreItemsForTimeline(currentInstance, timeline) } else { // user hasn't scrolled to the top, show a header instead this.store.setForTimeline(currentInstance, timeline, {shouldShowHeader: true}) } stop('handleItemIdsToAdd') } this.observe('itemIdsToAdd', (newItemIdsToAdd, oldItemIdsToAdd) => { if (!newItemIdsToAdd || !newItemIdsToAdd.length || isEqual(newItemIdsToAdd, oldItemIdsToAdd)) { return } scheduleIdleTask(handleItemIdsToAdd) }) }, setupFocus () { this.onPushState = this.onPushState.bind(this) this.store.setForCurrentTimeline({ ignoreBlurEvents: false }) window.addEventListener('pushState', this.onPushState) }, teardownFocus () { window.removeEventListener('pushState', this.onPushState) }, onPushState () { this.store.setForCurrentTimeline({ ignoreBlurEvents: true }) }, saveFocus (e) { try { let { currentInstance } = this.store.get() let { timeline } = this.get() let lastFocusedElementSelector let activeElement = e.target if (activeElement) { let focusKey = activeElement.getAttribute('focus-key') if (focusKey) { lastFocusedElementSelector = `[focus-key=${JSON.stringify(focusKey)}]` } } console.log('saving focus to ', lastFocusedElementSelector) this.store.setForTimeline(currentInstance, timeline, { lastFocusedElementSelector }) } catch (err) { console.error('unable to save focus', err) } }, clearFocus () { try { let { ignoreBlurEvents } = this.store.get() if (ignoreBlurEvents) { return } console.log('clearing focus') let { currentInstance } = this.store.get() let { timeline } = this.get() this.store.setForTimeline(currentInstance, timeline, { lastFocusedElementSelector: null }) } catch (err) { console.error('unable to clear focus', err) } }, restoreFocus () { let { lastFocusedElementSelector } = this.store.get() if (!lastFocusedElementSelector) { return } console.log('restoreFocus', lastFocusedElementSelector) requestAnimationFrame(() => { requestAnimationFrame(() => { let element = document.querySelector(lastFocusedElementSelector) if (element) { element.focus() } }) }) }, onNoNeedToScroll () { // If the timeline doesn't need to scroll, then we can safely "preinitialize," // i.e. render anything above the fold of the timeline. This avoids the affect // where the scrollable content appears to jump around if we need to scroll it. console.log('timeline preinitialized') this.store.set({timelinePreinitialized: true}) } } } </script>