tooot/src/components/GracefullyImage.tsx

140 lines
3.5 KiB
TypeScript
Raw Normal View History

2021-01-16 00:00:31 +01:00
import { StyleConstants } from '@utils/styles/constants'
2021-02-14 00:27:21 +01:00
import { useTheme } from '@utils/styles/ThemeManager'
import React, { useCallback, useMemo, useRef, useState } from 'react'
import {
Image,
ImageStyle,
Pressable,
StyleProp,
StyleSheet,
ViewStyle
} from 'react-native'
2021-01-28 01:31:19 +01:00
import { Blurhash } from 'react-native-blurhash'
2021-02-14 00:27:21 +01:00
// blurhas -> if blurhash, show before any loading succeed
// original -> load original
// original, remote -> if original failed, then remote
// preview, original -> first show preview, then original
// preview, original, remote -> first show preview, then original, if original failed, then remote
2021-01-16 00:00:31 +01:00
export interface Props {
hidden?: boolean
2021-03-19 21:44:52 +01:00
uri: { preview?: string; original?: string; remote?: string }
2021-01-16 00:00:31 +01:00
blurhash?: string
dimension?: { width: number; height: number }
onPress?: () => void
style?: StyleProp<ViewStyle>
2021-01-23 02:41:50 +01:00
imageStyle?: StyleProp<ImageStyle>
2021-02-14 00:27:21 +01:00
// For image viewer when there is no image size available
setImageDimensions?: React.Dispatch<
React.SetStateAction<{
width: number
height: number
}>
>
2021-01-16 00:00:31 +01:00
}
2021-02-02 22:50:38 +01:00
const GracefullyImage = React.memo(
({
hidden = false,
uri,
blurhash,
dimension,
onPress,
style,
2021-02-14 00:27:21 +01:00
imageStyle,
setImageDimensions
2021-02-02 22:50:38 +01:00
}: Props) => {
2021-02-14 00:27:21 +01:00
const { theme } = useTheme()
const originalFailed = useRef(false)
const [imageLoaded, setImageLoaded] = useState(false)
2021-01-16 00:00:31 +01:00
2021-02-14 00:27:21 +01:00
const source = useMemo(() => {
if (originalFailed.current) {
return { uri: uri.remote || undefined }
2021-02-02 22:50:38 +01:00
} else {
2021-02-14 00:27:21 +01:00
return { uri: uri.original }
2021-02-02 22:50:38 +01:00
}
2021-02-14 00:27:21 +01:00
}, [originalFailed.current])
const onLoad = useCallback(
({ nativeEvent }) => {
setImageLoaded(true)
setImageDimensions &&
setImageDimensions({
width: nativeEvent.width,
height: nativeEvent.height
})
},
[source.uri]
)
2021-02-02 22:50:38 +01:00
const onError = useCallback(() => {
2021-02-14 00:27:21 +01:00
if (!originalFailed.current) {
originalFailed.current = true
2021-02-02 22:50:38 +01:00
}
2021-02-14 00:27:21 +01:00
}, [originalFailed.current])
2021-02-02 22:50:38 +01:00
2021-02-14 00:27:21 +01:00
const previewView = useMemo(
() =>
uri.preview && !imageLoaded ? (
<Image
2021-02-14 00:27:21 +01:00
source={{ uri: uri.preview }}
style={[{ flex: 1 }, imageStyle]}
2021-02-10 00:40:44 +01:00
/>
2021-02-14 00:27:21 +01:00
) : null,
[imageLoaded]
)
const originalView = useMemo(
() => (
<Image
2021-02-14 00:27:21 +01:00
source={source}
style={[{ flex: 1 }, imageStyle]}
2021-02-14 00:27:21 +01:00
onLoad={onLoad}
onError={onError}
/>
),
[source]
2021-02-14 00:27:21 +01:00
)
const blurhashView = useMemo(() => {
return blurhash && (hidden || !imageLoaded) ? (
<Blurhash decodeAsync blurhash={blurhash} style={styles.blurhash} />
) : null
}, [hidden, imageLoaded])
2021-01-16 00:00:31 +01:00
2021-02-02 22:50:38 +01:00
return (
<Pressable
2021-03-21 23:06:53 +01:00
style={[
style,
dimension,
{ backgroundColor: theme.backgroundOverlayDefault }
]}
2021-02-02 22:50:38 +01:00
{...(onPress
? hidden
? { disabled: true }
: { onPress }
: { disabled: true })}
2021-02-14 00:27:21 +01:00
>
{previewView}
{originalView}
{blurhashView}
</Pressable>
2021-02-02 22:50:38 +01:00
)
},
2021-03-02 01:17:06 +01:00
(prev, next) =>
prev.hidden === next.hidden &&
prev.uri.preview === next.uri.preview &&
prev.uri.original === next.uri.original &&
prev.uri.remote === next.uri.remote
2021-02-02 22:50:38 +01:00
)
2021-01-16 00:00:31 +01:00
const styles = StyleSheet.create({
2021-02-14 00:27:21 +01:00
blurhash: {
width: '100%',
height: '100%',
position: 'absolute',
top: StyleConstants.Spacing.XS / 2,
left: StyleConstants.Spacing.XS / 2
2021-01-16 00:00:31 +01:00
}
})
export default GracefullyImage