1
0
mirror of https://github.com/Sonarr/Sonarr.git synced 2025-01-25 11:13:39 +02:00

Convert series images to TypeScript

This commit is contained in:
Mark McDowall 2024-09-02 16:05:19 -07:00 committed by Mark McDowall
parent e1e10e195c
commit ee99c3895d
7 changed files with 177 additions and 257 deletions

View File

@ -19,8 +19,10 @@ export type SeriesStatus = 'continuing' | 'ended' | 'upcoming' | 'deleted';
export type MonitorNewItems = 'all' | 'none';
export type CoverType = 'poster' | 'banner' | 'fanart' | 'season';
export interface Image {
coverType: string;
coverType: CoverType;
url: string;
remoteUrl: string;
}

View File

@ -1,29 +0,0 @@
import PropTypes from 'prop-types';
import React from 'react';
import SeriesImage from './SeriesImage';
const bannerPlaceholder = '';
function SeriesBanner(props) {
return (
<SeriesImage
{...props}
coverType="banner"
placeholder={bannerPlaceholder}
/>
);
}
SeriesBanner.propTypes = {
...SeriesImage.propTypes,
coverType: PropTypes.string,
placeholder: PropTypes.string,
overflow: PropTypes.bool,
size: PropTypes.number.isRequired
};
SeriesBanner.defaultProps = {
size: 70
};
export default SeriesBanner;

View File

@ -0,0 +1,23 @@
import React from 'react';
import SeriesImage, { SeriesImageProps } from './SeriesImage';
const bannerPlaceholder =
'';
interface SeriesBannerProps
extends Omit<SeriesImageProps, 'coverType' | 'placeholder'> {
size?: 35 | 70;
}
function SeriesBanner({ size = 70, ...otherProps }: SeriesBannerProps) {
return (
<SeriesImage
{...otherProps}
size={size}
coverType="banner"
placeholder={bannerPlaceholder}
/>
);
}
export default SeriesBanner;

View File

@ -1,198 +0,0 @@
import PropTypes from 'prop-types';
import React, { Component } from 'react';
import LazyLoad from 'react-lazyload';
function findImage(images, coverType) {
return images.find((image) => image.coverType === coverType);
}
function getUrl(image, coverType, size) {
const imageUrl = image?.url;
if (imageUrl) {
return imageUrl.replace(`${coverType}.jpg`, `${coverType}-${size}.jpg`);
}
}
class SeriesImage extends Component {
//
// Lifecycle
constructor(props, context) {
super(props, context);
const pixelRatio = Math.max(Math.round(window.devicePixelRatio), 1);
const {
images,
coverType,
size
} = props;
const image = findImage(images, coverType);
this.state = {
pixelRatio,
image,
url: getUrl(image, coverType, pixelRatio * size),
isLoaded: false,
hasError: false
};
}
componentDidMount() {
if (!this.state.url && this.props.onError) {
this.props.onError();
}
}
componentDidUpdate() {
const {
images,
coverType,
placeholder,
size,
onError
} = this.props;
const {
image,
pixelRatio
} = this.state;
const nextImage = findImage(images, coverType);
if (nextImage && (!image || nextImage.url !== image.url)) {
this.setState({
image: nextImage,
url: getUrl(nextImage, coverType, 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 (!nextImage && image) {
this.setState({
image: nextImage,
url: placeholder,
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,
placeholder,
size,
lazy,
overflow
} = this.props;
const {
url,
hasError,
isLoaded
} = this.state;
if (hasError || !url) {
return (
<img
className={className}
style={style}
src={placeholder}
/>
);
}
if (lazy) {
return (
<LazyLoad
height={size}
offset={100}
overflow={overflow}
placeholder={
<img
className={className}
style={style}
src={placeholder}
/>
}
>
<img
className={className}
style={style}
src={url}
onError={this.onError}
onLoad={this.onLoad}
rel="noreferrer"
/>
</LazyLoad>
);
}
return (
<img
className={className}
style={style}
src={isLoaded ? url : placeholder}
onError={this.onError}
onLoad={this.onLoad}
/>
);
}
}
SeriesImage.propTypes = {
className: PropTypes.string,
style: PropTypes.object,
images: PropTypes.arrayOf(PropTypes.object).isRequired,
coverType: PropTypes.string.isRequired,
placeholder: PropTypes.string.isRequired,
size: PropTypes.number.isRequired,
lazy: PropTypes.bool.isRequired,
overflow: PropTypes.bool.isRequired,
onError: PropTypes.func,
onLoad: PropTypes.func
};
SeriesImage.defaultProps = {
size: 250,
lazy: true,
overflow: false
};
export default SeriesImage;

View File

@ -0,0 +1,128 @@
import React, { useCallback, useEffect, useRef, useState } from 'react';
import LazyLoad from 'react-lazyload';
import { CoverType, Image } from './Series';
function findImage(images: Image[], coverType: CoverType) {
return images.find((image) => image.coverType === coverType);
}
function getUrl(image: Image, coverType: CoverType, size: number) {
const imageUrl = image?.url;
return imageUrl
? imageUrl.replace(`${coverType}.jpg`, `${coverType}-${size}.jpg`)
: null;
}
export interface SeriesImageProps {
className?: string;
style?: object;
images: Image[];
coverType: CoverType;
placeholder: string;
size?: number;
lazy?: boolean;
overflow?: boolean;
onError?: () => void;
onLoad?: () => void;
}
const pixelRatio = Math.max(Math.round(window.devicePixelRatio), 1);
function SeriesImage({
className,
style,
images,
coverType,
placeholder,
size = 250,
lazy = true,
overflow = false,
onError,
onLoad,
}: SeriesImageProps) {
const [url, setUrl] = useState<string | null>(null);
const [hasError, setHasError] = useState(false);
const [isLoaded, setIsLoaded] = useState(false);
const image = useRef<Image | null>(null);
const handleLoad = useCallback(() => {
setHasError(false);
setIsLoaded(true);
onLoad?.();
}, [setHasError, setIsLoaded, onLoad]);
const handleError = useCallback(() => {
setHasError(true);
setIsLoaded(false);
onError?.();
}, [setHasError, setIsLoaded, onError]);
useEffect(() => {
const nextImage = findImage(images, coverType);
if (nextImage && (!image.current || nextImage.url !== image.current.url)) {
// 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.
image.current = nextImage;
setUrl(getUrl(nextImage, coverType, pixelRatio * size));
setHasError(false);
} else if (!nextImage) {
if (image.current) {
image.current = null;
setUrl(placeholder);
setHasError(false);
onError?.();
}
}
}, [images, coverType, placeholder, size, onError]);
useEffect(() => {
if (!image.current) {
onError?.();
}
// This should only run once when the component mounts,
// so we don't need to include the other dependencies.
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);
if (hasError || !url) {
return <img className={className} style={style} src={placeholder} />;
}
if (lazy) {
return (
<LazyLoad
height={size}
offset={100}
overflow={overflow}
placeholder={
<img className={className} style={style} src={placeholder} />
}
>
<img
className={className}
style={style}
src={url}
rel="noreferrer"
onError={handleError}
onLoad={handleLoad}
/>
</LazyLoad>
);
}
return (
<img
className={className}
style={style}
src={isLoaded ? url : placeholder}
onError={handleError}
onLoad={handleLoad}
/>
);
}
export default SeriesImage;

View File

@ -1,29 +0,0 @@
import PropTypes from 'prop-types';
import React from 'react';
import SeriesImage from './SeriesImage';
const posterPlaceholder = '';
function SeriesPoster(props) {
return (
<SeriesImage
{...props}
coverType="poster"
placeholder={posterPlaceholder}
/>
);
}
SeriesPoster.propTypes = {
...SeriesImage.propTypes,
coverType: PropTypes.string,
placeholder: PropTypes.string,
overflow: PropTypes.bool,
size: PropTypes.number.isRequired
};
SeriesPoster.defaultProps = {
size: 250
};
export default SeriesPoster;

View File

@ -0,0 +1,23 @@
import React from 'react';
import SeriesImage, { SeriesImageProps } from './SeriesImage';
const posterPlaceholder =
'';
interface SeriesPosterProps
extends Omit<SeriesImageProps, 'coverType' | 'placeholder'> {
size?: 250 | 500;
}
function SeriesPoster({ size = 250, ...otherProps }: SeriesPosterProps) {
return (
<SeriesImage
{...otherProps}
size={size}
coverType="poster"
placeholder={posterPlaceholder}
/>
);
}
export default SeriesPoster;