import _ from 'lodash'; import PropTypes from 'prop-types'; import React, { Component } from 'react'; import LazyLoad from 'react-lazyload'; const coverPlaceholder = ''; function findCover(images) { return _.find(images, { coverType: 'cover' }); } function getCoverUrl(cover, size) { if (cover) { if (cover.url.contains('lastWrite=') || (/^https?:/).test(cover.url)) { // Remove protocol let url = cover.url.replace(/^https?:/, ''); url = url.replace('cover.jpg', `cover-${size}.jpg`); return url; } } } class AlbumCover extends Component { // // Lifecycle constructor(props, context) { super(props, context); const pixelRatio = Math.floor(window.devicePixelRatio); const { images, size } = props; const cover = findCover(images); this.state = { pixelRatio, cover, coverUrl: getCoverUrl(cover, pixelRatio * size), isLoaded: false, hasError: false }; } componentDidUpdate(prevProps) { const { images, size } = this.props; const { cover, pixelRatio } = this.state; const nextcover = findCover(images); if (nextcover && (!cover || nextcover.url !== cover.url)) { this.setState({ cover: nextcover, coverUrl: getCoverUrl(nextcover, 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. }); } } // // Listeners onError = () => { this.setState({ hasError: true }); } onLoad = () => { this.setState({ isLoaded: true, hasError: false }); } // // Render render() { const { className, style, size, lazy, overflow } = this.props; const { coverUrl, hasError, isLoaded } = this.state; if (hasError || !coverUrl) { return ( ); } if (lazy) { return ( } > ); } return ( ); } } AlbumCover.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 }; AlbumCover.defaultProps = { size: 250, lazy: true, overflow: false }; export default AlbumCover;