147 lines
2.9 KiB
JavaScript
147 lines
2.9 KiB
JavaScript
import React from 'react'
|
|
import PropTypes from 'prop-types'
|
|
import { connect } from 'react-redux'
|
|
import noop from 'lodash.noop'
|
|
import {
|
|
fetchBundleRequest,
|
|
fetchBundleSuccess,
|
|
fetchBundleFail,
|
|
} from '../../../actions/bundles'
|
|
|
|
const emptyComponent = () => null
|
|
|
|
class Bundle extends React.PureComponent {
|
|
|
|
static cache = new Map
|
|
|
|
state = {
|
|
mod: undefined,
|
|
forceRender: false,
|
|
}
|
|
|
|
componentWillMount() {
|
|
this.load(this.props)
|
|
}
|
|
|
|
componentWillReceiveProps(nextProps) {
|
|
if (nextProps.fetchComponent !== this.props.fetchComponent) {
|
|
this.load(nextProps)
|
|
}
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
if (this.timeout) {
|
|
clearTimeout(this.timeout)
|
|
}
|
|
}
|
|
|
|
load = (props) => {
|
|
const {
|
|
fetchComponent,
|
|
onFetch,
|
|
onFetchSuccess,
|
|
onFetchFail,
|
|
renderDelay
|
|
} = props || this.props
|
|
|
|
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)
|
|
})
|
|
}
|
|
|
|
render() {
|
|
const {
|
|
loading: LoadingComponent,
|
|
error: ErrorComponent,
|
|
children,
|
|
renderDelay
|
|
} = this.props
|
|
|
|
const { mod, forceRender } = this.state
|
|
const elapsed = this.timestamp ? (new Date() - this.timestamp) : renderDelay
|
|
|
|
if (mod === undefined) {
|
|
return (elapsed >= renderDelay || forceRender) ? <LoadingComponent /> : null
|
|
} else if (mod === null) {
|
|
return <ErrorComponent onRetry={this.load} />
|
|
}
|
|
|
|
return children(mod)
|
|
}
|
|
|
|
}
|
|
|
|
const mapDispatchToProps = (dispatch) => ({
|
|
onFetch() {
|
|
dispatch(fetchBundleRequest())
|
|
},
|
|
onFetchSuccess() {
|
|
dispatch(fetchBundleSuccess())
|
|
},
|
|
onFetchFail(error) {
|
|
dispatch(fetchBundleFail(error))
|
|
},
|
|
})
|
|
|
|
Bundle.propTypes = {
|
|
fetchComponent: PropTypes.func,
|
|
loading: PropTypes.func,
|
|
error: PropTypes.func,
|
|
children: PropTypes.func.isRequired,
|
|
renderDelay: PropTypes.number,
|
|
onFetch: PropTypes.func,
|
|
onFetchSuccess: PropTypes.func,
|
|
onFetchFail: PropTypes.func,
|
|
}
|
|
|
|
Bundle.defaultProps = {
|
|
loading: emptyComponent,
|
|
error: emptyComponent,
|
|
renderDelay: 0,
|
|
onFetch: noop,
|
|
onFetchSuccess: noop,
|
|
onFetchFail: noop,
|
|
}
|
|
|
|
export default connect(null, mapDispatchToProps)(Bundle) |