UNPKG

rs-react-native-image-gallery

Version:
92 lines 15.1 kB
import { memo, useCallback, useEffect, useMemo, useState } from 'react'; import { ActivityIndicator, Dimensions, Image, StyleSheet, Text, TouchableWithoutFeedback, View } from 'react-native'; import { LOADING_TIMEOUT } from './constants'; // ---------------------------------------------------------------------- // Get dimensions outside component to avoid recreating on every render var _a = Dimensions.get('window'), initialHeight = _a.height, initialWidth = _a.width; // Cache for tracking already loaded images to avoid showing the loading indicator again var loadedImages = new Set(); var ImagePreview = memo(function (_a) { var index = _a.index, isSelected = _a.isSelected, item = _a.item, renderCustomImage = _a.renderCustomImage, resizeMode = _a.resizeMode; // Create a unique identifier for the image for caching var imageKey = item.url || (typeof item.source === 'object' && 'uri' in item.source ? item.source.uri : String(item.source)); var _b = useState(!loadedImages.has(imageKey || '')), isLoading = _b[0], setIsLoading = _b[1]; var _c = useState(false), hasError = _c[0], setHasError = _c[1]; // Use dimensions from closure or through hook if dynamic resizing is needed var _d = useMemo(function () { return ({ height: initialHeight, width: initialWidth }); }, []), height = _d.height, width = _d.width; // Create memoized styles with dynamic dimensions var containerStyle = useMemo(function () { return ({ height: height, width: width }); }, [height, width]); var handleLoadEnd = useCallback(function () { setIsLoading(false); if (imageKey) { loadedImages.add(imageKey); } }, [imageKey]); var handleError = useCallback(function () { setHasError(true); setIsLoading(false); }, []); // Add loading timeout to prevent infinite spinner useEffect(function () { if (!isLoading) return; var timeoutId = setTimeout(function () { if (isLoading) { setIsLoading(false); } }, LOADING_TIMEOUT); return function () { return clearTimeout(timeoutId); }; }, [isLoading]); return (<View style={{ width: '100%', height: '100%' }}> <TouchableWithoutFeedback> <View style={[containerStyle, { overflow: 'hidden' }]}> {renderCustomImage ? (renderCustomImage(item, index, isSelected)) : (<Image resizeMode={resizeMode} source={item.source || { uri: item.url }} // @ts-ignore style={styles.image} onLoadStart={function () { setIsLoading(true); setHasError(false); }} onLoad={handleLoadEnd} onError={handleError} accessible={true} accessibilityLabel={"Image ".concat(index + 1)} fadeDuration={300}/>)} {isLoading && <ActivityIndicator style={styles.loader} color="white" size="large"/>} {hasError && (<View style={styles.errorContainer}> <Text style={styles.errorText}>Failed to load image</Text> </View>)} </View> </TouchableWithoutFeedback> </View>); }, function (prevProps, nextProps) { // Only re-render if the selected index changes or the image URL changes return (prevProps.isSelected === nextProps.isSelected && prevProps.item.url === nextProps.item.url && prevProps.item.source === nextProps.item.source && prevProps.index === nextProps.index); }); var styles = StyleSheet.create({ image: { height: '100%', width: '100%' }, loader: { position: 'absolute', top: '50%', left: '50%', transform: [{ translateX: -15 }, { translateY: -15 }] }, errorContainer: { position: 'absolute', top: '50%', left: '50%', transform: [{ translateX: -75 }, { translateY: -15 }], backgroundColor: 'rgba(0, 0, 0, 0.7)', padding: 10, borderRadius: 5 }, errorText: { color: 'white', fontSize: 16 } }); export default ImagePreview; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image-preview.js","sourceRoot":"","sources":["../src/image-preview.tsx"],"names":[],"mappings":"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","sourcesContent":["import { memo, useCallback, useEffect, useMemo, useState } from 'react';\nimport { ActivityIndicator, Dimensions, Image, StyleSheet, Text, TouchableWithoutFeedback, View } from 'react-native';\nimport { LOADING_TIMEOUT } from './constants';\nimport type { ImagePreviewProps } from './types';\n\n// ----------------------------------------------------------------------\n\n// Get dimensions outside component to avoid recreating on every render\nconst { height: initialHeight, width: initialWidth } = Dimensions.get('window');\n\n// Cache for tracking already loaded images to avoid showing the loading indicator again\nconst loadedImages = new Set<string>();\n\nconst ImagePreview = memo(\n\t({ index, isSelected, item, renderCustomImage, resizeMode }: ImagePreviewProps) => {\n\t\t// Create a unique identifier for the image for caching\n\t\tconst imageKey = item.url || (typeof item.source === 'object' && 'uri' in item.source ? item.source.uri : String(item.source));\n\t\tconst [isLoading, setIsLoading] = useState(!loadedImages.has(imageKey || ''));\n\t\tconst [hasError, setHasError] = useState(false);\n\t\t// Use dimensions from closure or through hook if dynamic resizing is needed\n\t\tconst { height, width } = useMemo(() => ({ height: initialHeight, width: initialWidth }), []);\n\n\t\t// Create memoized styles with dynamic dimensions\n\t\tconst containerStyle = useMemo(\n\t\t\t() => ({\n\t\t\t\theight,\n\t\t\t\twidth\n\t\t\t}),\n\t\t\t[height, width]\n\t\t);\n\n\t\tconst handleLoadEnd = useCallback(() => {\n\t\t\tsetIsLoading(false);\n\t\t\tif (imageKey) {\n\t\t\t\tloadedImages.add(imageKey);\n\t\t\t}\n\t\t}, [imageKey]);\n\n\t\tconst handleError = useCallback(() => {\n\t\t\tsetHasError(true);\n\t\t\tsetIsLoading(false);\n\t\t}, []);\n\n\t\t// Add loading timeout to prevent infinite spinner\n\t\tuseEffect(() => {\n\t\t\tif (!isLoading) return;\n\n\t\t\tconst timeoutId = setTimeout(() => {\n\t\t\t\tif (isLoading) {\n\t\t\t\t\tsetIsLoading(false);\n\t\t\t\t}\n\t\t\t}, LOADING_TIMEOUT);\n\n\t\t\treturn () => clearTimeout(timeoutId);\n\t\t}, [isLoading]);\n\n\t\treturn (\n\t\t\t<View style={{ width: '100%', height: '100%' }}>\n\t\t\t\t<TouchableWithoutFeedback>\n\t\t\t\t\t<View style={[containerStyle, { overflow: 'hidden' }]}>\n\t\t\t\t\t\t{renderCustomImage ? (\n\t\t\t\t\t\t\trenderCustomImage(item, index, isSelected)\n\t\t\t\t\t\t) : (\n\t\t\t\t\t\t\t<Image\n\t\t\t\t\t\t\t\tresizeMode={resizeMode}\n\t\t\t\t\t\t\t\tsource={item.source || { uri: item.url }}\n\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\tstyle={styles.image}\n\t\t\t\t\t\t\t\tonLoadStart={() => {\n\t\t\t\t\t\t\t\t\tsetIsLoading(true);\n\t\t\t\t\t\t\t\t\tsetHasError(false);\n\t\t\t\t\t\t\t\t}}\n\t\t\t\t\t\t\t\tonLoad={handleLoadEnd}\n\t\t\t\t\t\t\t\tonError={handleError}\n\t\t\t\t\t\t\t\taccessible={true}\n\t\t\t\t\t\t\t\taccessibilityLabel={`Image ${index + 1}`}\n\t\t\t\t\t\t\t\tfadeDuration={300}\n\t\t\t\t\t\t\t/>\n\t\t\t\t\t\t)}\n\t\t\t\t\t\t{isLoading && <ActivityIndicator style={styles.loader} color=\"white\" size=\"large\" />}\n\t\t\t\t\t\t{hasError && (\n\t\t\t\t\t\t\t<View style={styles.errorContainer}>\n\t\t\t\t\t\t\t\t<Text style={styles.errorText}>Failed to load image</Text>\n\t\t\t\t\t\t\t</View>\n\t\t\t\t\t\t)}\n\t\t\t\t\t</View>\n\t\t\t\t</TouchableWithoutFeedback>\n\t\t\t</View>\n\t\t);\n\t},\n\t(prevProps, nextProps) => {\n\t\t// Only re-render if the selected index changes or the image URL changes\n\t\treturn (\n\t\t\tprevProps.isSelected === nextProps.isSelected &&\n\t\t\tprevProps.item.url === nextProps.item.url &&\n\t\t\tprevProps.item.source === nextProps.item.source &&\n\t\t\tprevProps.index === nextProps.index\n\t\t);\n\t}\n);\n\nconst styles = StyleSheet.create({\n\timage: {\n\t\theight: '100%',\n\t\twidth: '100%'\n\t},\n\tloader: {\n\t\tposition: 'absolute',\n\t\ttop: '50%',\n\t\tleft: '50%',\n\t\ttransform: [{ translateX: -15 }, { translateY: -15 }]\n\t},\n\terrorContainer: {\n\t\tposition: 'absolute',\n\t\ttop: '50%',\n\t\tleft: '50%',\n\t\ttransform: [{ translateX: -75 }, { translateY: -15 }],\n\t\tbackgroundColor: 'rgba(0, 0, 0, 0.7)',\n\t\tpadding: 10,\n\t\tborderRadius: 5\n\t},\n\terrorText: {\n\t\tcolor: 'white',\n\t\tfontSize: 16\n\t}\n});\n\nexport default ImagePreview;\n"]}