import _ from 'lodash'; import PropTypes from 'prop-types'; import React, { Component } from 'react'; import LazyLoad from 'react-lazyload'; const posterPlaceholder = ''; function findPoster(images) { return _.find(images, { coverType: 'poster' }); } function getPosterUrl(poster, size) { if (poster) { // Remove protocol let url = poster.url.replace(/^https?:/, ''); url = url.replace('poster.jpg', `poster-${size}.jpg`); return url; } } class MoviePoster extends Component { // // Lifecycle constructor(props, context) { super(props, context); const pixelRatio = Math.ceil(window.devicePixelRatio); const { images, size } = props; const poster = findPoster(images); this.state = { pixelRatio, poster, posterUrl: getPosterUrl(poster, pixelRatio * size), isLoaded: false, hasError: false }; } componentDidMount() { if (!this.state.posterUrl && this.props.onError) { this.props.onError(); } } componentDidUpdate(prevProps, prevState) { const { images, size, onError } = this.props; const { poster, pixelRatio } = this.state; const nextPoster = findPoster(images); if (nextPoster && (!poster || nextPoster.url !== poster.url)) { this.setState({ poster: nextPoster, posterUrl: getPosterUrl(nextPoster, pixelRatio * size), hasError: false // Don't reset isLoaded, as we want to immediately try to // show the new image, whether an image was shown previously // or the placeholder was shown. }); } else if (!nextPoster && poster) { this.setState({ poster: nextPoster, posterUrl: posterPlaceholder, hasError: false }); if (onError) { onError(); } } } // // Listeners onError = () => { this.setState({ hasError: true }); if (this.props.onError) { this.props.onError(); } } onLoad = () => { this.setState({ isLoaded: true, hasError: false }); if (this.props.onLoad) { this.props.onLoad(); } } // // Render render() { const { className, style, size, lazy, overflow } = this.props; const { posterUrl, hasError, isLoaded } = this.state; if (hasError || !posterUrl) { return ( ); } if (lazy) { return ( } > ); } return ( ); } } MoviePoster.propTypes = { className: PropTypes.string, style: PropTypes.object, images: PropTypes.arrayOf(PropTypes.object).isRequired, size: PropTypes.number.isRequired, lazy: PropTypes.bool.isRequired, overflow: PropTypes.bool.isRequired, onError: PropTypes.func, onLoad: PropTypes.func }; MoviePoster.defaultProps = { size: 250, lazy: true, overflow: false }; export default MoviePoster;