rs-react-native-image-gallery
Version:
React Native Image Gallery with Thumbnails
214 lines (210 loc) • 35.8 kB
JavaScript
import { FlashList } from '@shopify/flash-list';
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
import { Dimensions, Image, StyleSheet, Text, TouchableOpacity, View } from 'react-native';
import Modal from 'react-native-modal';
import { preLoadImages } from './_helpers';
import { DEFAULT_THUMB_COLOR, DEFAULT_THUMB_SIZE, THUMB_SPACING } from './constants';
import ImagePreview from './image-preview';
import SwipeContainer from './swipe-container';
// ----------------------------------------------------------------------
var _a = Dimensions.get('window'), deviceHeight = _a.height, deviceWidth = _a.width;
var ImageGallery = function (_a) {
var close = _a.close, _b = _a.hideThumbs, hideThumbs = _b === void 0 ? false : _b, _c = _a.images, images = _c === void 0 ? [] : _c, _d = _a.initialIndex, initialIndex = _d === void 0 ? 0 : _d, isOpen = _a.isOpen, renderCustomImage = _a.renderCustomImage, renderCustomThumb = _a.renderCustomThumb, renderFooterComponent = _a.renderFooterComponent, renderHeaderComponent = _a.renderHeaderComponent, _e = _a.resizeMode, resizeMode = _e === void 0 ? 'contain' : _e, _f = _a.thumbColor, thumbColor = _f === void 0 ? DEFAULT_THUMB_COLOR : _f, _g = _a.thumbResizeMode, thumbResizeMode = _g === void 0 ? 'cover' : _g, _h = _a.thumbSize, thumbSize = _h === void 0 ? DEFAULT_THUMB_SIZE : _h, _j = _a.disableSwipe, disableSwipe = _j === void 0 ? false : _j;
var _k = useState(initialIndex), activeIndex = _k[0], setActiveIndex = _k[1];
var _l = useState(false), isDragging = _l[0], setIsDragging = _l[1];
var topRef = useRef(null);
var bottomRef = useRef(null);
// Memoize sizes to prevent unnecessary re-renders
var mainListSize = useMemo(function () { return ({ width: deviceWidth, height: deviceHeight }); }, []);
var keyExtractor = useCallback(function (item, index) { var _a, _b; return (_b = (_a = item.id) === null || _a === void 0 ? void 0 : _a.toString()) !== null && _b !== void 0 ? _b : index.toString(); }, []);
var scrollToIndex = useCallback(function (index) {
if (!isOpen || index < 0 || index >= images.length)
return;
// Update active index first
setActiveIndex(index);
// Create scroll options - center the item in the view
var scrollOptions = {
animated: true,
index: index,
viewPosition: 0.5
};
// Use setTimeout to ensure animations work properly with state update
setTimeout(function () {
try {
if (topRef.current) {
topRef.current.scrollToIndex(scrollOptions);
}
if (bottomRef.current && !hideThumbs) {
bottomRef.current.scrollToIndex(scrollOptions);
}
}
catch (error) {
console.warn('Error scrolling to index:', error);
}
}, 0);
}, [hideThumbs, isOpen, images.length]);
var renderItem = useCallback(function (_a) {
var item = _a.item, index = _a.index;
return (<View style={{ width: deviceWidth, height: deviceHeight, flex: 1 }}>
<ImagePreview index={index} isSelected={activeIndex === index} item={item} resizeMode={resizeMode} renderCustomImage={renderCustomImage}/>
</View>);
}, [activeIndex, resizeMode, renderCustomImage, deviceWidth, deviceHeight]);
var renderThumb = function (_a) {
var item = _a.item, index = _a.index;
// Check active state on each render
var isActive = activeIndex === index;
var imageSource = item.thumbSource || item.source || { uri: item.thumbUrl || item.url };
// Create proper conditional styling that won't fail with null/undefined
var thumbStyle;
if (isActive) {
thumbStyle = [
styles.thumb,
{
width: thumbSize,
height: thumbSize,
borderWidth: 3,
borderColor: thumbColor
}
];
}
else {
thumbStyle = [styles.thumb, { width: thumbSize, height: thumbSize }];
}
return (<TouchableOpacity onPress={function () { return scrollToIndex(index); }} activeOpacity={0.8} style={{
height: thumbSize,
marginRight: THUMB_SPACING,
justifyContent: 'center',
alignItems: 'center'
}}>
{renderCustomThumb ? (renderCustomThumb(item, index, isActive)) : (<Image resizeMode={thumbResizeMode} style={thumbStyle} source={imageSource} fadeDuration={100}/>)}
</TouchableOpacity>);
};
var onMomentumEnd = useCallback(function (e) {
var x = e.nativeEvent.contentOffset.x;
var newIndex = Math.round(x / deviceWidth);
// Only update if different to avoid unnecessary re-renders
if (newIndex !== activeIndex) {
scrollToIndex(newIndex);
}
}, [deviceWidth, scrollToIndex, activeIndex]);
useEffect(function () {
// Only set the initial index when opening the gallery
if (isOpen && initialIndex >= 0 && initialIndex < images.length) {
setActiveIndex(initialIndex);
}
else if (!isOpen) {
// Reset the active index when closing, but don't scroll
setActiveIndex(0);
}
}, [isOpen, initialIndex, images.length]);
// Preload image thumbnails when gallery opens
useEffect(function () {
if (isOpen && images.length > 0) {
var urls = images.map(function (img) { return img.thumbUrl || img.url; }).filter(Boolean);
preLoadImages(urls);
}
}, [isOpen, images]);
return (<Modal isVisible={isOpen} onBackdropPress={close} onBackButtonPress={close} onSwipeComplete={close} swipeDirection={disableSwipe ? [] : ['down']} style={{ margin: 0, justifyContent: 'flex-end' }} useNativeDriverForBackdrop={true} useNativeDriver={true} hideModalContentWhileAnimating={true} propagateSwipe={true} onModalHide={function () {
if (bottomRef.current) {
bottomRef.current.scrollToIndex({
index: activeIndex,
animated: false,
viewPosition: 0.5
});
}
if (topRef.current) {
topRef.current.scrollToIndex({
index: activeIndex,
animated: false,
viewPosition: 0.5
});
}
}} onModalWillShow={function () {
if (bottomRef.current) {
bottomRef.current.scrollToIndex({
index: activeIndex,
animated: false,
viewPosition: 0.5
});
}
if (topRef.current) {
topRef.current.scrollToIndex({
index: activeIndex,
animated: false,
viewPosition: 0.5
});
}
}}>
<View style={styles.container}>
{images.length > 0 && (<>
<SwipeContainer disableSwipe={disableSwipe} setIsDragging={setIsDragging} close={close}>
<FlashList initialScrollIndex={initialIndex < images.length ? initialIndex : 0} estimatedItemSize={deviceWidth} data={images} horizontal keyExtractor={keyExtractor} onMomentumScrollEnd={onMomentumEnd} pagingEnabled ref={topRef} renderItem={renderItem} scrollEnabled={!isDragging} showsHorizontalScrollIndicator={false} estimatedListSize={mainListSize} drawDistance={deviceWidth} removeClippedSubviews={true}/>
</SwipeContainer>
{!hideThumbs && (<View style={styles.bottomFlatlist}>
<FlashList initialScrollIndex={initialIndex < images.length ? initialIndex : 0} estimatedItemSize={thumbSize + THUMB_SPACING} contentContainerStyle={styles.thumbnailListContainer} data={images} horizontal keyExtractor={keyExtractor} pagingEnabled={false} ref={bottomRef} renderItem={renderThumb} showsHorizontalScrollIndicator={false} drawDistance={deviceWidth * 2} removeClippedSubviews={true} extraData={activeIndex} overrideItemLayout={function (layout, _item, _index) {
layout.size = thumbSize + THUMB_SPACING;
}}/>
</View>)}
</>)}
{/* Page indicator */}
{images.length > 0 && (<View style={styles.pageIndicator}>
<Text style={styles.pageIndicatorText}>{"".concat(activeIndex + 1, " / ").concat(images.length)}</Text>
</View>)}
{renderHeaderComponent && images.length > 0 && (<View style={styles.header}>{renderHeaderComponent(images[activeIndex], activeIndex)}</View>)}
{renderFooterComponent && images.length > 0 && (<View style={styles.footer}>{renderFooterComponent(images[activeIndex], activeIndex)}</View>)}
</View>
</Modal>);
};
var styles = StyleSheet.create({
container: {
alignItems: 'center',
backgroundColor: 'black',
flex: 1,
height: deviceHeight,
justifyContent: 'center',
width: deviceWidth
},
header: {
position: 'absolute',
top: 0,
width: '100%'
},
footer: {
position: 'absolute',
bottom: 0,
width: '100%'
},
activeThumb: {
borderWidth: 3
},
thumb: {
borderRadius: 12
},
thumbnailListContainer: {
paddingVertical: 20,
paddingHorizontal: 10
},
bottomFlatlist: {
position: 'absolute',
width: deviceWidth,
height: DEFAULT_THUMB_SIZE * 1.5,
bottom: DEFAULT_THUMB_SIZE,
left: 0
},
pageIndicator: {
position: 'absolute',
top: 15,
alignSelf: 'center',
backgroundColor: 'rgba(0, 0, 0, 0.5)',
padding: 8,
borderRadius: 15,
zIndex: 20
},
pageIndicatorText: {
color: 'white',
fontSize: 14,
fontWeight: 'bold'
}
});
export default ImageGallery;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image-gallery.js","sourceRoot":"","sources":["../src/image-gallery.tsx"],"names":[],"mappings":"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","sourcesContent":["import { FlashList } from '@shopify/flash-list';\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react';\nimport { Dimensions, Image, StyleSheet, Text, TouchableOpacity, View } from 'react-native';\nimport Modal from 'react-native-modal';\nimport { preLoadImages } from './_helpers';\nimport { DEFAULT_THUMB_COLOR, DEFAULT_THUMB_SIZE, THUMB_SPACING } from './constants';\nimport ImagePreview from './image-preview';\nimport SwipeContainer from './swipe-container';\nimport type { GalleryProps, ImageObject, RenderImageProps } from './types';\n\n// ----------------------------------------------------------------------\n\nconst { height: deviceHeight, width: deviceWidth } = Dimensions.get('window');\n\nconst ImageGallery: React.FC<GalleryProps> = ({\n\tclose,\n\thideThumbs = false,\n\timages = [],\n\tinitialIndex = 0,\n\tisOpen,\n\trenderCustomImage,\n\trenderCustomThumb,\n\trenderFooterComponent,\n\trenderHeaderComponent,\n\tresizeMode = 'contain',\n\tthumbColor = DEFAULT_THUMB_COLOR,\n\tthumbResizeMode = 'cover',\n\tthumbSize = DEFAULT_THUMB_SIZE,\n\tdisableSwipe = false\n}) => {\n\tconst [activeIndex, setActiveIndex] = useState(initialIndex);\n\tconst [isDragging, setIsDragging] = useState(false);\n\n\tconst topRef = useRef<FlashList<ImageObject>>(null);\n\tconst bottomRef = useRef<FlashList<ImageObject>>(null);\n\n\t// Memoize sizes to prevent unnecessary re-renders\n\tconst mainListSize = useMemo(() => ({ width: deviceWidth, height: deviceHeight }), []);\n\n\tconst keyExtractor = useCallback((item: ImageObject, index: number) => item.id?.toString() ?? index.toString(), []);\n\n\tconst scrollToIndex = useCallback(\n\t\t(index: number) => {\n\t\t\tif (!isOpen || index < 0 || index >= images.length) return;\n\n\t\t\t// Update active index first\n\t\t\tsetActiveIndex(index);\n\n\t\t\t// Create scroll options - center the item in the view\n\t\t\tconst scrollOptions = {\n\t\t\t\tanimated: true,\n\t\t\t\tindex,\n\t\t\t\tviewPosition: 0.5\n\t\t\t};\n\n\t\t\t// Use setTimeout to ensure animations work properly with state update\n\t\t\tsetTimeout(() => {\n\t\t\t\ttry {\n\t\t\t\t\tif (topRef.current) {\n\t\t\t\t\t\ttopRef.current.scrollToIndex(scrollOptions);\n\t\t\t\t\t}\n\n\t\t\t\t\tif (bottomRef.current && !hideThumbs) {\n\t\t\t\t\t\tbottomRef.current.scrollToIndex(scrollOptions);\n\t\t\t\t\t}\n\t\t\t\t} catch (error) {\n\t\t\t\t\tconsole.warn('Error scrolling to index:', error);\n\t\t\t\t}\n\t\t\t}, 0);\n\t\t},\n\t\t[hideThumbs, isOpen, images.length]\n\t);\n\n\tconst renderItem = useCallback(\n\t\t({ item, index }: RenderImageProps) => (\n\t\t\t<View style={{ width: deviceWidth, height: deviceHeight, flex: 1 }}>\n\t\t\t\t<ImagePreview\n\t\t\t\t\tindex={index}\n\t\t\t\t\tisSelected={activeIndex === index}\n\t\t\t\t\titem={item}\n\t\t\t\t\tresizeMode={resizeMode}\n\t\t\t\t\trenderCustomImage={renderCustomImage}\n\t\t\t\t/>\n\t\t\t</View>\n\t\t),\n\t\t[activeIndex, resizeMode, renderCustomImage, deviceWidth, deviceHeight]\n\t);\n\n\tconst renderThumb = ({ item, index }: RenderImageProps) => {\n\t\t// Check active state on each render\n\t\tconst isActive = activeIndex === index;\n\t\tconst imageSource = item.thumbSource || item.source || { uri: item.thumbUrl || item.url };\n\n\t\t// Create proper conditional styling that won't fail with null/undefined\n\t\tlet thumbStyle;\n\t\tif (isActive) {\n\t\t\tthumbStyle = [\n\t\t\t\tstyles.thumb,\n\t\t\t\t{\n\t\t\t\t\twidth: thumbSize,\n\t\t\t\t\theight: thumbSize,\n\t\t\t\t\tborderWidth: 3,\n\t\t\t\t\tborderColor: thumbColor\n\t\t\t\t}\n\t\t\t];\n\t\t} else {\n\t\t\tthumbStyle = [styles.thumb, { width: thumbSize, height: thumbSize }];\n\t\t}\n\n\t\treturn (\n\t\t\t<TouchableOpacity\n\t\t\t\tonPress={() => scrollToIndex(index)}\n\t\t\t\tactiveOpacity={0.8}\n\t\t\t\tstyle={{\n\t\t\t\t\theight: thumbSize,\n\t\t\t\t\tmarginRight: THUMB_SPACING,\n\t\t\t\t\tjustifyContent: 'center',\n\t\t\t\t\talignItems: 'center'\n\t\t\t\t}}>\n\t\t\t\t{renderCustomThumb ? (\n\t\t\t\t\trenderCustomThumb(item, index, isActive)\n\t\t\t\t) : (\n\t\t\t\t\t<Image resizeMode={thumbResizeMode} style={thumbStyle} source={imageSource} fadeDuration={100} />\n\t\t\t\t)}\n\t\t\t</TouchableOpacity>\n\t\t);\n\t};\n\n\tconst onMomentumEnd = useCallback(\n\t\t(e: { nativeEvent: { contentOffset: { x: number } } }) => {\n\t\t\tconst { x } = e.nativeEvent.contentOffset;\n\t\t\tconst newIndex = Math.round(x / deviceWidth);\n\n\t\t\t// Only update if different to avoid unnecessary re-renders\n\t\t\tif (newIndex !== activeIndex) {\n\t\t\t\tscrollToIndex(newIndex);\n\t\t\t}\n\t\t},\n\t\t[deviceWidth, scrollToIndex, activeIndex]\n\t);\n\n\tuseEffect(() => {\n\t\t// Only set the initial index when opening the gallery\n\t\tif (isOpen && initialIndex >= 0 && initialIndex < images.length) {\n\t\t\tsetActiveIndex(initialIndex);\n\t\t} else if (!isOpen) {\n\t\t\t// Reset the active index when closing, but don't scroll\n\t\t\tsetActiveIndex(0);\n\t\t}\n\t}, [isOpen, initialIndex, images.length]);\n\n\t// Preload image thumbnails when gallery opens\n\tuseEffect(() => {\n\t\tif (isOpen && images.length > 0) {\n\t\t\tconst urls = images.map((img) => img.thumbUrl || img.url).filter(Boolean);\n\t\t\tpreLoadImages(urls);\n\t\t}\n\t}, [isOpen, images]);\n\n\treturn (\n\t\t<Modal\n\t\t\tisVisible={isOpen}\n\t\t\tonBackdropPress={close}\n\t\t\tonBackButtonPress={close}\n\t\t\tonSwipeComplete={close}\n\t\t\tswipeDirection={disableSwipe ? [] : ['down']}\n\t\t\tstyle={{ margin: 0, justifyContent: 'flex-end' }}\n\t\t\tuseNativeDriverForBackdrop={true}\n\t\t\tuseNativeDriver={true}\n\t\t\thideModalContentWhileAnimating={true}\n\t\t\tpropagateSwipe={true}\n\t\t\tonModalHide={() => {\n\t\t\t\tif (bottomRef.current) {\n\t\t\t\t\tbottomRef.current.scrollToIndex({\n\t\t\t\t\t\tindex: activeIndex,\n\t\t\t\t\t\tanimated: false,\n\t\t\t\t\t\tviewPosition: 0.5\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t\tif (topRef.current) {\n\t\t\t\t\ttopRef.current.scrollToIndex({\n\t\t\t\t\t\tindex: activeIndex,\n\t\t\t\t\t\tanimated: false,\n\t\t\t\t\t\tviewPosition: 0.5\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t}}\n\t\t\tonModalWillShow={() => {\n\t\t\t\tif (bottomRef.current) {\n\t\t\t\t\tbottomRef.current.scrollToIndex({\n\t\t\t\t\t\tindex: activeIndex,\n\t\t\t\t\t\tanimated: false,\n\t\t\t\t\t\tviewPosition: 0.5\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t\tif (topRef.current) {\n\t\t\t\t\ttopRef.current.scrollToIndex({\n\t\t\t\t\t\tindex: activeIndex,\n\t\t\t\t\t\tanimated: false,\n\t\t\t\t\t\tviewPosition: 0.5\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t}}>\n\t\t\t<View style={styles.container}>\n\t\t\t\t{images.length > 0 && (\n\t\t\t\t\t<>\n\t\t\t\t\t\t<SwipeContainer disableSwipe={disableSwipe} setIsDragging={setIsDragging} close={close}>\n\t\t\t\t\t\t\t<FlashList\n\t\t\t\t\t\t\t\tinitialScrollIndex={initialIndex < images.length ? initialIndex : 0}\n\t\t\t\t\t\t\t\testimatedItemSize={deviceWidth}\n\t\t\t\t\t\t\t\tdata={images}\n\t\t\t\t\t\t\t\thorizontal\n\t\t\t\t\t\t\t\tkeyExtractor={keyExtractor}\n\t\t\t\t\t\t\t\tonMomentumScrollEnd={onMomentumEnd}\n\t\t\t\t\t\t\t\tpagingEnabled\n\t\t\t\t\t\t\t\tref={topRef}\n\t\t\t\t\t\t\t\trenderItem={renderItem}\n\t\t\t\t\t\t\t\tscrollEnabled={!isDragging}\n\t\t\t\t\t\t\t\tshowsHorizontalScrollIndicator={false}\n\t\t\t\t\t\t\t\testimatedListSize={mainListSize}\n\t\t\t\t\t\t\t\tdrawDistance={deviceWidth}\n\t\t\t\t\t\t\t\tremoveClippedSubviews={true}\n\t\t\t\t\t\t\t/>\n\t\t\t\t\t\t</SwipeContainer>\n\n\t\t\t\t\t\t{!hideThumbs && (\n\t\t\t\t\t\t\t<View style={styles.bottomFlatlist}>\n\t\t\t\t\t\t\t\t<FlashList\n\t\t\t\t\t\t\t\t\tinitialScrollIndex={initialIndex < images.length ? initialIndex : 0}\n\t\t\t\t\t\t\t\t\testimatedItemSize={thumbSize + THUMB_SPACING}\n\t\t\t\t\t\t\t\t\tcontentContainerStyle={styles.thumbnailListContainer}\n\t\t\t\t\t\t\t\t\tdata={images}\n\t\t\t\t\t\t\t\t\thorizontal\n\t\t\t\t\t\t\t\t\tkeyExtractor={keyExtractor}\n\t\t\t\t\t\t\t\t\tpagingEnabled={false}\n\t\t\t\t\t\t\t\t\tref={bottomRef}\n\t\t\t\t\t\t\t\t\trenderItem={renderThumb}\n\t\t\t\t\t\t\t\t\tshowsHorizontalScrollIndicator={false}\n\t\t\t\t\t\t\t\t\tdrawDistance={deviceWidth * 2}\n\t\t\t\t\t\t\t\t\tremoveClippedSubviews={true}\n\t\t\t\t\t\t\t\t\textraData={activeIndex}\n\t\t\t\t\t\t\t\t\toverrideItemLayout={(layout, _item, _index) => {\n\t\t\t\t\t\t\t\t\t\tlayout.size = thumbSize + THUMB_SPACING;\n\t\t\t\t\t\t\t\t\t}}\n\t\t\t\t\t\t\t\t/>\n\t\t\t\t\t\t\t</View>\n\t\t\t\t\t\t)}\n\t\t\t\t\t</>\n\t\t\t\t)}\n\n\t\t\t\t{/* Page indicator */}\n\t\t\t\t{images.length > 0 && (\n\t\t\t\t\t<View style={styles.pageIndicator}>\n\t\t\t\t\t\t<Text style={styles.pageIndicatorText}>{`${activeIndex + 1} / ${images.length}`}</Text>\n\t\t\t\t\t</View>\n\t\t\t\t)}\n\n\t\t\t\t{renderHeaderComponent && images.length > 0 && (\n\t\t\t\t\t<View style={styles.header}>{renderHeaderComponent(images[activeIndex], activeIndex)}</View>\n\t\t\t\t)}\n\n\t\t\t\t{renderFooterComponent && images.length > 0 && (\n\t\t\t\t\t<View style={styles.footer}>{renderFooterComponent(images[activeIndex], activeIndex)}</View>\n\t\t\t\t)}\n\t\t\t</View>\n\t\t</Modal>\n\t);\n};\n\nconst styles = StyleSheet.create({\n\tcontainer: {\n\t\talignItems: 'center',\n\t\tbackgroundColor: 'black',\n\t\tflex: 1,\n\t\theight: deviceHeight,\n\t\tjustifyContent: 'center',\n\t\twidth: deviceWidth\n\t},\n\theader: {\n\t\tposition: 'absolute',\n\t\ttop: 0,\n\t\twidth: '100%'\n\t},\n\tfooter: {\n\t\tposition: 'absolute',\n\t\tbottom: 0,\n\t\twidth: '100%'\n\t},\n\tactiveThumb: {\n\t\tborderWidth: 3\n\t},\n\tthumb: {\n\t\tborderRadius: 12\n\t},\n\tthumbnailListContainer: {\n\t\tpaddingVertical: 20,\n\t\tpaddingHorizontal: 10\n\t},\n\tbottomFlatlist: {\n\t\tposition: 'absolute',\n\t\twidth: deviceWidth,\n\t\theight: DEFAULT_THUMB_SIZE * 1.5,\n\t\tbottom: DEFAULT_THUMB_SIZE,\n\t\tleft: 0\n\t},\n\tpageIndicator: {\n\t\tposition: 'absolute',\n\t\ttop: 15,\n\t\talignSelf: 'center',\n\t\tbackgroundColor: 'rgba(0, 0, 0, 0.5)',\n\t\tpadding: 8,\n\t\tborderRadius: 15,\n\t\tzIndex: 20\n\t},\n\tpageIndicatorText: {\n\t\tcolor: 'white',\n\t\tfontSize: 14,\n\t\tfontWeight: 'bold'\n\t}\n});\n\nexport default ImageGallery;\n"]}