Lidarr/frontend/src/Artist/ArtistBanner.js

176 lines
6.0 KiB
JavaScript
Raw Normal View History

2017-09-17 06:24:15 +00:00
import _ from 'lodash';
import PropTypes from 'prop-types';
import React, { Component } from 'react';
import LazyLoad from 'react-lazyload';
const bannerPlaceholder = '';
function findBanner(images) {
return _.find(images, { coverType: 'banner' });
}
function getBannerUrl(banner, size) {
if (banner) {
if (banner.url.contains('lastWrite=') || (/^https?:/).test(banner.url)) {
2017-09-17 06:24:15 +00:00
// Remove protocol
let url = banner.url.replace(/^https?:/, '');
url = url.replace('banner.jpg', `banner-${size}.jpg`);
2017-09-17 06:24:15 +00:00
return url;
}
2017-09-17 06:24:15 +00:00
}
}
class ArtistBanner extends Component {
//
// Lifecycle
constructor(props, context) {
super(props, context);
const pixelRatio = Math.floor(window.devicePixelRatio);
const {
images,
size
} = props;
const banner = findBanner(images);
this.state = {
pixelRatio,
banner,
bannerUrl: getBannerUrl(banner, pixelRatio * size),
isLoaded: false,
hasError: false
2017-09-17 06:24:15 +00:00
};
}
componentDidUpdate(prevProps) {
const {
images,
size
} = this.props;
const {
banner,
2017-09-17 06:24:15 +00:00
pixelRatio
} = this.state;
const nextBanner = findBanner(images);
2017-09-17 06:24:15 +00:00
if (nextBanner && (!banner || nextBanner.url !== banner.url)) {
2017-09-17 06:24:15 +00:00
this.setState({
banner: nextBanner,
bannerUrl: getBannerUrl(nextBanner, pixelRatio * size),
hasError: false,
isLoaded: true
2017-09-17 06:24:15 +00:00
});
}
// The banner could not be loaded..
if (!nextBanner && (this.props !== prevProps)) {
this.setState({
banner: undefined,
bannerUrl: bannerPlaceholder,
hasError: true
});
2017-09-17 06:24:15 +00:00
}
}
//
// Listeners
onError = () => {
this.setState({ hasError: true });
}
onLoad = () => {
this.setState({
isLoaded: true,
hasError: false
});
2017-09-17 06:24:15 +00:00
}
//
// Render
render() {
const {
className,
style,
size,
lazy,
overflow
} = this.props;
const {
bannerUrl,
hasError,
isLoaded
} = this.state;
if (hasError || !bannerUrl) {
return (
<img
className={className}
style={style}
src={bannerPlaceholder}
/>
);
}
if (lazy) {
return (
<LazyLoad
height={size}
offset={100}
overflow={overflow}
placeholder={
<img
className={className}
style={style}
src={bannerPlaceholder}
/>
}
>
<img
className={className}
style={style}
src={bannerUrl}
onError={this.onError}
/>
</LazyLoad>
);
}
return (
<img
className={className}
style={style}
src={isLoaded ? bannerUrl : bannerPlaceholder}
onError={this.onError}
onLoad={this.onLoad}
/>
);
}
}
ArtistBanner.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
};
ArtistBanner.defaultProps = {
size: 70,
lazy: true,
overflow: false
};
export default ArtistBanner;