2021-07-14 22:14:04 +08:00
|
|
|
// @flow
|
|
|
|
import React from 'react';
|
2022-03-21 12:20:33 -04:00
|
|
|
import { getImageProxyUrl, getThumbnailCdnUrl } from 'util/thumbnail';
|
2021-07-14 22:14:04 +08:00
|
|
|
|
2022-03-21 11:40:32 +08:00
|
|
|
function scaleToDevicePixelRatio(value: number) {
|
2021-07-14 22:14:04 +08:00
|
|
|
const devicePixelRatio = window.devicePixelRatio || 1.0;
|
2022-04-01 09:04:12 +08:00
|
|
|
const nextInteger = Math.ceil(value * devicePixelRatio);
|
|
|
|
// Round to next 100px for better caching
|
|
|
|
return Math.ceil(nextInteger / 100) * 100;
|
2021-07-14 22:14:04 +08:00
|
|
|
}
|
|
|
|
|
2022-03-21 11:40:32 +08:00
|
|
|
function getOptimizedImgUrl(url, width, height, quality) {
|
|
|
|
let optimizedUrl = url;
|
|
|
|
if (url && !url.startsWith('/public/')) {
|
|
|
|
optimizedUrl = url.trim().replace(/^http:\/\//i, 'https://');
|
2021-07-14 22:14:04 +08:00
|
|
|
|
2022-03-21 12:20:33 -04:00
|
|
|
if (optimizedUrl.endsWith('.gif')) {
|
|
|
|
optimizedUrl = getImageProxyUrl(optimizedUrl);
|
|
|
|
} else {
|
2022-03-21 11:40:32 +08:00
|
|
|
optimizedUrl = getThumbnailCdnUrl({ thumbnail: optimizedUrl, width, height, quality });
|
2021-07-14 22:14:04 +08:00
|
|
|
}
|
|
|
|
}
|
2022-03-21 11:40:32 +08:00
|
|
|
return optimizedUrl;
|
|
|
|
}
|
2021-07-14 22:14:04 +08:00
|
|
|
|
2022-03-21 11:40:32 +08:00
|
|
|
// ****************************************************************************
|
|
|
|
// OptimizedImage
|
|
|
|
// ****************************************************************************
|
2021-07-14 22:14:04 +08:00
|
|
|
|
2022-03-21 11:40:32 +08:00
|
|
|
type Props = {
|
|
|
|
src: string,
|
|
|
|
width?: number,
|
|
|
|
quality?: number,
|
|
|
|
waitLoad?: boolean,
|
|
|
|
};
|
2021-07-14 22:14:04 +08:00
|
|
|
|
2022-03-21 11:40:32 +08:00
|
|
|
function OptimizedImage(props: Props) {
|
|
|
|
const {
|
|
|
|
src,
|
|
|
|
width = 0, // 0 = use intrinsic width
|
2022-04-01 21:50:10 -04:00
|
|
|
quality = 95,
|
2022-03-21 11:40:32 +08:00
|
|
|
waitLoad,
|
|
|
|
...imgProps
|
|
|
|
} = props;
|
2021-07-14 22:14:04 +08:00
|
|
|
|
2022-03-21 11:40:32 +08:00
|
|
|
const ref = React.useRef<any>();
|
|
|
|
const optimizedSrc = getOptimizedImgUrl(src, scaleToDevicePixelRatio(width), 0, quality);
|
2021-07-14 22:14:04 +08:00
|
|
|
|
2022-02-07 12:59:20 -08:00
|
|
|
if (!src || !optimizedSrc) {
|
2021-07-14 22:14:04 +08:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<img
|
|
|
|
ref={ref}
|
|
|
|
{...imgProps}
|
2021-10-27 15:20:47 -03:00
|
|
|
style={{ visibility: waitLoad ? 'hidden' : 'visible' }}
|
2021-07-14 22:14:04 +08:00
|
|
|
src={optimizedSrc}
|
2021-10-27 15:20:47 -03:00
|
|
|
onLoad={() => {
|
2022-03-21 11:40:32 +08:00
|
|
|
if (waitLoad) {
|
|
|
|
ref.current.style.visibility = 'visible';
|
|
|
|
}
|
2021-10-27 15:20:47 -03:00
|
|
|
}}
|
2021-07-14 22:14:04 +08:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export default OptimizedImage;
|