bigbuffet-rw/app/soapbox/features/ui/components/bundle.tsx

115 lines
2.6 KiB
TypeScript
Raw Normal View History

import React from 'react';
2020-03-27 13:59:38 -07:00
const emptyComponent = () => null;
const noop = () => { };
export interface BundleProps {
fetchComponent: () => Promise<any>
loading: React.ComponentType
error: React.ComponentType<{ onRetry: (props?: BundleProps) => void }>
children: (mod: any) => React.ReactNode
renderDelay?: number
onFetch: () => void
onFetchSuccess: () => void
onFetchFail: (error: any) => void
2022-08-31 15:22:37 -07:00
}
interface BundleState {
mod: any
forceRender: boolean
2022-08-31 15:22:37 -07:00
}
/** Fetches and renders an async component. */
class Bundle extends React.PureComponent<BundleProps, BundleState> {
timeout: NodeJS.Timeout | undefined;
timestamp: Date | undefined;
2020-03-27 13:59:38 -07:00
static defaultProps = {
loading: emptyComponent,
error: emptyComponent,
renderDelay: 0,
onFetch: noop,
onFetchSuccess: noop,
onFetchFail: noop,
2023-01-05 09:55:08 -08:00
};
2020-03-27 13:59:38 -07:00
2023-01-05 09:55:08 -08:00
static cache = new Map;
2020-03-27 13:59:38 -07:00
state = {
mod: undefined,
forceRender: false,
2023-01-05 09:55:08 -08:00
};
2020-03-27 13:59:38 -07:00
componentDidMount() {
this.load(this.props);
2020-03-27 13:59:38 -07:00
}
UNSAFE_componentWillReceiveProps(nextProps: BundleProps) {
if (nextProps.fetchComponent !== this.props.fetchComponent) {
this.load(nextProps);
2020-03-27 13:59:38 -07:00
}
}
componentWillUnmount() {
2020-03-27 13:59:38 -07:00
if (this.timeout) {
clearTimeout(this.timeout);
}
}
load = (props?: BundleProps) => {
const { fetchComponent, onFetch, onFetchSuccess, onFetchFail, renderDelay } = props || this.props;
2020-03-27 13:59:38 -07:00
const cachedMod = Bundle.cache.get(fetchComponent);
if (fetchComponent === undefined) {
this.setState({ mod: null });
return Promise.resolve();
}
onFetch();
if (cachedMod) {
this.setState({ mod: cachedMod.default });
onFetchSuccess();
return Promise.resolve();
}
this.setState({ mod: undefined });
if (renderDelay !== 0) {
this.timestamp = new Date();
this.timeout = setTimeout(() => this.setState({ forceRender: true }), renderDelay);
}
return fetchComponent()
.then((mod) => {
Bundle.cache.set(fetchComponent, mod);
this.setState({ mod: mod.default });
onFetchSuccess();
})
.catch((error) => {
this.setState({ mod: null });
onFetchFail(error);
});
2023-01-05 09:55:08 -08:00
};
2020-03-27 13:59:38 -07:00
render() {
const { loading: Loading, error: Error, children, renderDelay } = this.props;
const { mod, forceRender } = this.state;
2022-08-31 15:22:37 -07:00
const elapsed = this.timestamp ? ((new Date()).getTime() - this.timestamp.getTime()) : renderDelay!;
2020-03-27 13:59:38 -07:00
if (mod === undefined) {
2022-08-31 15:22:37 -07:00
return (elapsed >= renderDelay! || forceRender) ? <Loading /> : null;
2020-03-27 13:59:38 -07:00
}
if (mod === null) {
return <Error onRetry={this.load} />;
}
return children(mod);
}
}
export default Bundle;