// @flow import * as React from 'react'; import {readData} from "../../utils/WebData"; import i18n from "i18n-js"; import {Snackbar} from 'react-native-paper'; import {RefreshControl, SectionList, View} from "react-native"; import NetworkErrorComponent from "../Custom/NetworkErrorComponent"; import BasicLoadingScreen from "../Custom/BasicLoadingScreen"; type Props = { navigation: Object, fetchUrl: string, autoRefreshTime: number, refreshOnFocus: boolean, renderItem: React.Node, renderSectionHeader: React.Node, stickyHeader: boolean, createDataset: Function, updateData: number, } type State = { refreshing: boolean, firstLoading: boolean, fetchedData: ?Object, snackbarVisible: boolean }; const MIN_REFRESH_TIME = 5 * 1000; /** * Component used to render a SectionList with data fetched from the web * * This is a pure component, meaning it will only update if a shallow comparison of state and props is different. * To force the component to update, change the value of updateData. */ export default class WebSectionList extends React.PureComponent { static defaultProps = { renderSectionHeader: null, stickyHeader: false, updateData: 0, }; refreshInterval: IntervalID; lastRefresh: Date; state = { refreshing: false, firstLoading: true, fetchedData: undefined, snackbarVisible: false }; onRefresh: Function; onFetchSuccess: Function; onFetchError: Function; getEmptySectionHeader: Function; showSnackBar: Function; hideSnackBar: Function; constructor() { super(); // creating references to functions used in render() this.onRefresh = this.onRefresh.bind(this); this.onFetchSuccess = this.onFetchSuccess.bind(this); this.onFetchError = this.onFetchError.bind(this); this.getEmptySectionHeader = this.getEmptySectionHeader.bind(this); this.showSnackBar = this.showSnackBar.bind(this); this.hideSnackBar = this.hideSnackBar.bind(this); } /** * Registers react navigation events on first screen load. * Allows to detect when the screen is focused */ componentDidMount() { const onScreenFocus = this.onScreenFocus.bind(this); const onScreenBlur = this.onScreenBlur.bind(this); this.props.navigation.addListener('focus', onScreenFocus); this.props.navigation.addListener('blur', onScreenBlur); this.onRefresh(); } /** * Refreshes data when focusing the screen and setup a refresh interval if asked to */ onScreenFocus() { if (this.props.refreshOnFocus && this.lastRefresh !== undefined) this.onRefresh(); if (this.props.autoRefreshTime > 0) this.refreshInterval = setInterval(this.onRefresh, this.props.autoRefreshTime) } /** * Removes any interval on un-focus */ onScreenBlur() { clearInterval(this.refreshInterval); } /** * Callback used when fetch is successful. * It will update the displayed data and stop the refresh animation * * @param fetchedData The newly fetched data */ onFetchSuccess(fetchedData: Object) { this.setState({ fetchedData: fetchedData, refreshing: false, firstLoading: false }); this.lastRefresh = new Date(); } /** * Callback used when fetch encountered an error. * It will reset the displayed data and show an error. */ onFetchError() { this.setState({ fetchedData: undefined, refreshing: false, firstLoading: false }); this.showSnackBar(); } /** * Refreshes data and shows an animations while doing it */ onRefresh() { let canRefresh; if (this.lastRefresh !== undefined) canRefresh = (new Date().getTime() - this.lastRefresh.getTime()) > MIN_REFRESH_TIME; else canRefresh = true; if (canRefresh) { this.setState({refreshing: true}); readData(this.props.fetchUrl) .then(this.onFetchSuccess) .catch(this.onFetchError); } } /** * Gets an empty section header * * @param section The current section * @return {*} */ getEmptySectionHeader({section}: Object) { return ; } /** * Shows the error popup */ showSnackBar() { this.setState({snackbarVisible: true}) } /** * Hides the error popup */ hideSnackBar() { this.setState({snackbarVisible: false}) } render() { let dataset = []; if (this.state.fetchedData !== undefined) dataset = this.props.createDataset(this.state.fetchedData); const shouldRenderHeader = this.props.renderSectionHeader !== null; return ( {i18n.t("homeScreen.listUpdateFail")} {/*$FlowFixMe*/} } //$FlowFixMe renderSectionHeader={shouldRenderHeader ? this.props.renderSectionHeader : this.getEmptySectionHeader} //$FlowFixMe renderItem={this.props.renderItem} stickySectionHeadersEnabled={this.props.stickyHeader} contentContainerStyle={{minHeight: '100%'}} style={{minHeight: '100%'}} ListEmptyComponent={this.state.refreshing ? : } /> ); } }