react-image-gallery
Version:
React carousel image gallery component with thumbnail and mobile support
1,218 lines (1,070 loc) • 36.2 kB
JSX
import React from 'react';
import Swipeable from 'react-swipeable';
import throttle from 'lodash.throttle';
import debounce from 'lodash.debounce';
import ResizeObserver from 'resize-observer-polyfill';
import PropTypes from 'prop-types';
const screenChangeEvents = [
'fullscreenchange',
'MSFullscreenChange',
'mozfullscreenchange',
'webkitfullscreenchange'
];
export default class ImageGallery extends React.Component {
constructor(props) {
super(props);
this.state = {
currentIndex: props.startIndex,
thumbsTranslate: 0,
offsetPercentage: 0,
galleryWidth: 0,
thumbnailsWrapperWidth: 0,
thumbnailsWrapperHeight: 0,
isFullscreen: false,
isPlaying: false
};
// Used to update the throttle if slideDuration changes
this._unthrottledSlideToIndex = this.slideToIndex;
this.slideToIndex = throttle(this._unthrottledSlideToIndex,
props.slideDuration,
{trailing: false});
if (props.lazyLoad) {
this._lazyLoaded = [];
}
}
static propTypes = {
flickThreshold: PropTypes.number,
items: PropTypes.array.isRequired,
showNav: PropTypes.bool,
autoPlay: PropTypes.bool,
lazyLoad: PropTypes.bool,
infinite: PropTypes.bool,
showIndex: PropTypes.bool,
showBullets: PropTypes.bool,
showThumbnails: PropTypes.bool,
showPlayButton: PropTypes.bool,
showFullscreenButton: PropTypes.bool,
disableThumbnailScroll: PropTypes.bool,
disableArrowKeys: PropTypes.bool,
disableSwipe: PropTypes.bool,
useBrowserFullscreen: PropTypes.bool,
preventDefaultTouchmoveEvent: PropTypes.bool,
defaultImage: PropTypes.string,
indexSeparator: PropTypes.string,
thumbnailPosition: PropTypes.string,
startIndex: PropTypes.number,
slideDuration: PropTypes.number,
slideInterval: PropTypes.number,
slideOnThumbnailOver: PropTypes.bool,
swipeThreshold: PropTypes.number,
swipingTransitionDuration: PropTypes.number,
onSlide: PropTypes.func,
onScreenChange: PropTypes.func,
onPause: PropTypes.func,
onPlay: PropTypes.func,
onClick: PropTypes.func,
onImageLoad: PropTypes.func,
onImageError: PropTypes.func,
onTouchMove: PropTypes.func,
onTouchEnd: PropTypes.func,
onTouchStart: PropTypes.func,
onMouseOver: PropTypes.func,
onMouseLeave: PropTypes.func,
onThumbnailError: PropTypes.func,
onThumbnailClick: PropTypes.func,
renderCustomControls: PropTypes.func,
renderLeftNav: PropTypes.func,
renderRightNav: PropTypes.func,
renderPlayPauseButton: PropTypes.func,
renderFullscreenButton: PropTypes.func,
renderItem: PropTypes.func,
stopPropagation: PropTypes.bool,
additionalClass: PropTypes.string,
useTranslate3D: PropTypes.bool,
isRTL: PropTypes.bool
};
static defaultProps = {
items: [],
showNav: true,
autoPlay: false,
lazyLoad: false,
infinite: true,
showIndex: false,
showBullets: false,
showThumbnails: true,
showPlayButton: true,
showFullscreenButton: true,
disableThumbnailScroll: false,
disableArrowKeys: false,
disableSwipe: false,
useTranslate3D: true,
isRTL: false,
useBrowserFullscreen: true,
preventDefaultTouchmoveEvent: false,
flickThreshold: 0.4,
stopPropagation: false,
indexSeparator: ' / ',
thumbnailPosition: 'bottom',
startIndex: 0,
slideDuration: 450,
swipingTransitionDuration: 0,
slideInterval: 3000,
swipeThreshold: 30,
renderLeftNav: (onClick, disabled) => {
return (
<button
type='button'
className='image-gallery-left-nav'
disabled={disabled}
onClick={onClick}
aria-label='Previous Slide'
/>
);
},
renderRightNav: (onClick, disabled) => {
return (
<button
type='button'
className='image-gallery-right-nav'
disabled={disabled}
onClick={onClick}
aria-label='Next Slide'
/>
);
},
renderPlayPauseButton: (onClick, isPlaying) => {
return (
<button
type='button'
className={
`image-gallery-play-button${isPlaying ? ' active' : ''}`}
onClick={onClick}
aria-label='Play or Pause Slideshow'
/>
);
},
renderFullscreenButton: (onClick, isFullscreen) => {
return (
<button
type='button'
className={
`image-gallery-fullscreen-button${isFullscreen ? ' active' : ''}`}
onClick={onClick}
aria-label='Open Fullscreen'
/>
);
},
};
componentWillReceiveProps(nextProps) {
if (nextProps.lazyLoad &&
(!this.props.lazyLoad || this.props.items !== nextProps.items)) {
this._lazyLoaded = [];
}
}
componentDidUpdate(prevProps, prevState) {
const itemsChanged = prevProps.items.length !== this.props.items.length;
if (itemsChanged) {
this._handleResize();
}
if (prevState.currentIndex !== this.state.currentIndex) {
this._updateThumbnailTranslate(prevState.currentIndex);
}
if (prevProps.slideDuration !== this.props.slideDuration) {
this.slideToIndex = throttle(this._unthrottledSlideToIndex,
this.props.slideDuration,
{trailing: false});
}
}
componentDidMount() {
if (this.props.autoPlay) {
this.play();
}
window.addEventListener('keydown', this._handleKeyDown);
this._onScreenChangeEvent();
}
componentWillUnmount() {
window.removeEventListener('keydown', this._handleKeyDown);
this._offScreenChangeEvent();
if (this._intervalId) {
window.clearInterval(this._intervalId);
this._intervalId = null;
}
if(this.resizeObserver && this._imageGallerySlideWrapper) {
this.resizeObserver.unobserve(this._imageGallerySlideWrapper);
}
if (this._transitionTimer) {
window.clearTimeout(this._transitionTimer);
}
if (this._createResizeObserver) {
this._createResizeObserver();
}
}
play(callback = true) {
if (!this._intervalId) {
const {slideInterval, slideDuration} = this.props;
this.setState({isPlaying: true});
this._intervalId = window.setInterval(() => {
if (!this.props.infinite && !this._canSlideRight()) {
this.pause();
} else {
this.slideToIndex(this.state.currentIndex + 1);
}
}, Math.max(slideInterval, slideDuration));
if (this.props.onPlay && callback) {
this.props.onPlay(this.state.currentIndex);
}
}
}
pause(callback = true) {
if (this._intervalId) {
window.clearInterval(this._intervalId);
this._intervalId = null;
this.setState({isPlaying: false});
if (this.props.onPause && callback) {
this.props.onPause(this.state.currentIndex);
}
}
}
setModalFullscreen(state) {
this.setState({modalFullscreen: state});
// manually call because browser does not support screenchange events
if (this.props.onScreenChange) {
this.props.onScreenChange(state);
}
}
fullScreen() {
const gallery = this._imageGallery;
if (this.props.useBrowserFullscreen) {
if (gallery.requestFullscreen) {
gallery.requestFullscreen();
} else if (gallery.msRequestFullscreen) {
gallery.msRequestFullscreen();
} else if (gallery.mozRequestFullScreen) {
gallery.mozRequestFullScreen();
} else if (gallery.webkitRequestFullscreen) {
gallery.webkitRequestFullscreen();
} else {
// fallback to fullscreen modal for unsupported browsers
this.setModalFullscreen(true);
}
} else {
this.setModalFullscreen(true);
}
this.setState({isFullscreen: true});
}
exitFullScreen() {
if (this.state.isFullscreen) {
if (this.props.useBrowserFullscreen) {
if (document.exitFullscreen) {
document.exitFullscreen();
} else if (document.webkitExitFullscreen) {
document.webkitExitFullscreen();
} else if (document.mozCancelFullScreen) {
document.mozCancelFullScreen();
} else if (document.msExitFullscreen) {
document.msExitFullscreen();
} else {
// fallback to fullscreen modal for unsupported browsers
this.setModalFullscreen(false);
}
} else {
this.setModalFullscreen(false);
}
this.setState({isFullscreen: false});
}
}
slideToIndex = (index, event) => {
const {currentIndex, isTransitioning} = this.state;
if (!isTransitioning) {
if (event) {
if (this._intervalId) {
// user triggered event while ImageGallery is playing, reset interval
this.pause(false);
this.play(false);
}
}
let slideCount = this.props.items.length - 1;
let nextIndex = index;
if (index < 0) {
nextIndex = slideCount;
} else if (index > slideCount) {
nextIndex = 0;
}
this.setState({
previousIndex: currentIndex,
currentIndex: nextIndex,
isTransitioning: nextIndex !== currentIndex,
offsetPercentage: 0,
style: {
transition: `all ${this.props.slideDuration}ms ease-out`
}
}, this._onSliding);
}
};
_onSliding = () => {
const { isTransitioning } = this.state;
this._transitionTimer = window.setTimeout(() => {
if (isTransitioning) {
this.setState({isTransitioning: !isTransitioning});
if (this.props.onSlide) {
this.props.onSlide(this.state.currentIndex);
}
}
}, this.props.slideDuration + 50);
};
getCurrentIndex() {
return this.state.currentIndex;
}
_handleScreenChange = () => {
/*
handles screen change events that the browser triggers e.g. esc key
*/
const fullScreenElement = document.fullscreenElement ||
document.msFullscreenElement ||
document.mozFullScreenElement ||
document.webkitFullscreenElement;
if (this.props.onScreenChange) {
this.props.onScreenChange(fullScreenElement);
}
this.setState({isFullscreen: !!fullScreenElement});
};
_onScreenChangeEvent() {
screenChangeEvents.map(eventName => {
document.addEventListener(eventName, this._handleScreenChange);
});
}
_offScreenChangeEvent() {
screenChangeEvents.map(eventName => {
document.removeEventListener(eventName, this._handleScreenChange);
});
}
_toggleFullScreen = () => {
if (this.state.isFullscreen) {
this.exitFullScreen();
} else {
this.fullScreen();
}
};
_togglePlay = () => {
if (this._intervalId) {
this.pause();
} else {
this.play();
}
};
_initGalleryResizing = (element) => {
/*
When image-gallery-slide-wrapper unmounts and mounts when thumbnail bar position is changed
ref is called twice, once with null and another with the element.
Make sure element is available before calling observe.
*/
if (element) {
this._imageGallerySlideWrapper = element;
this.resizeObserver = new ResizeObserver(this._createResizeObserver);
this.resizeObserver.observe(element);
}
};
_createResizeObserver = debounce((entries) => {
if (!entries) return;
entries.forEach(() => {
this._handleResize();
});
}, 300);
_handleResize = () => {
const { currentIndex } = this.state;
if (this._imageGallery) {
this.setState({
galleryWidth: this._imageGallery.offsetWidth
});
}
if (this._imageGallerySlideWrapper) {
this.setState({
gallerySlideWrapperHeight: this._imageGallerySlideWrapper.offsetHeight
});
}
if (this._thumbnailsWrapper) {
if (this._isThumbnailVertical()) {
this.setState({thumbnailsWrapperHeight: this._thumbnailsWrapper.offsetHeight});
} else {
this.setState({thumbnailsWrapperWidth: this._thumbnailsWrapper.offsetWidth});
}
}
// Adjust thumbnail container when thumbnail width or height is adjusted
this._setThumbsTranslate(-this._getThumbsTranslate(currentIndex));
};
_isThumbnailVertical() {
const { thumbnailPosition } = this.props;
return thumbnailPosition === 'left' || thumbnailPosition === 'right';
}
_handleKeyDown = (event) => {
if (this.props.disableArrowKeys) {
return;
}
const LEFT_ARROW = 37;
const RIGHT_ARROW = 39;
const ESC_KEY = 27;
const key = parseInt(event.keyCode || event.which || 0);
switch(key) {
case LEFT_ARROW:
if (this._canSlideLeft() && !this._intervalId) {
this._slideLeft();
}
break;
case RIGHT_ARROW:
if (this._canSlideRight() && !this._intervalId) {
this._slideRight();
}
break;
case ESC_KEY:
if (this.state.isFullscreen && !this.props.useBrowserFullscreen) {
this.exitFullScreen();
}
}
};
_handleImageError = (event) => {
if (this.props.defaultImage &&
event.target.src.indexOf(this.props.defaultImage) === -1) {
event.target.src = this.props.defaultImage;
}
};
_setScrollDirection(deltaX, deltaY) {
const { scrollingUpDown, scrollingLeftRight } = this.state;
const x = Math.abs(deltaX);
const y = Math.abs(deltaY);
// If y > x the user is scrolling up and down
if (y > x && !scrollingUpDown && !scrollingLeftRight) {
this.setState({ scrollingUpDown: true });
} else if (!scrollingLeftRight && !scrollingUpDown) {
this.setState({ scrollingLeftRight: true });
}
};
_handleOnSwiped = (e, deltaX, deltaY, isFlick) => {
const { scrollingUpDown, scrollingLeftRight } = this.state;
const { isRTL } = this.props;
if (scrollingUpDown) {
// user stopped scrollingUpDown
this.setState({ scrollingUpDown: false });
}
if (scrollingLeftRight) {
// user stopped scrollingLeftRight
this.setState({ scrollingLeftRight: false });
}
if (!scrollingUpDown) { // don't swipe if user is scrolling
const side = (deltaX > 0 ? 1 : -1) * (isRTL ? -1 : 1);//if it is RTL the direction is reversed
this._handleOnSwipedTo(side, isFlick);
}
};
_handleOnSwipedTo(side, isFlick) {
const { currentIndex, isTransitioning } = this.state;
let slideTo = currentIndex;
if ((this._sufficientSwipeOffset() || isFlick) && !isTransitioning) {
slideTo += side;
}
if (side < 0) {
if (!this._canSlideLeft()) {
slideTo = currentIndex;
}
} else {
if (!this._canSlideRight()) {
slideTo = currentIndex;
}
}
this._unthrottledSlideToIndex(slideTo);
}
_sufficientSwipeOffset() {
return Math.abs(this.state.offsetPercentage) > this.props.swipeThreshold;
}
_handleSwiping = (e, deltaX, deltaY, delta) => {
const { galleryWidth, isTransitioning, scrollingUpDown } = this.state;
const { swipingTransitionDuration } = this.props;
this._setScrollDirection(deltaX, deltaY);
if (!isTransitioning && !scrollingUpDown) {
const side = deltaX < 0 ? 1 : -1;
let offsetPercentage = (delta / galleryWidth * 100);
if (Math.abs(offsetPercentage) >= 100) {
offsetPercentage = 100;
}
const swipingTransition = {
transition: `transform ${swipingTransitionDuration}ms ease-out`
};
this.setState({
offsetPercentage: side * offsetPercentage,
style: swipingTransition,
});
} else {
// don't move the slide
this.setState({ offsetPercentage: 0 });
}
};
_canNavigate() {
return this.props.items.length >= 2;
}
_canSlideLeft() {
return this.props.infinite ||
(this.props.isRTL ? this._canSlideNext() : this._canSlidePrevious());
}
_canSlideRight() {
return this.props.infinite ||
(this.props.isRTL ? this._canSlidePrevious() : this._canSlideNext());
}
_canSlidePrevious() {
return this.state.currentIndex > 0;
}
_canSlideNext() {
return this.state.currentIndex < this.props.items.length - 1;
}
_updateThumbnailTranslate(previousIndex) {
const { thumbsTranslate, currentIndex } = this.state;
if (this.state.currentIndex === 0) {
this._setThumbsTranslate(0);
} else {
let indexDifference = Math.abs(previousIndex - currentIndex);
let scroll = this._getThumbsTranslate(indexDifference);
if (scroll > 0) {
if (previousIndex < currentIndex) {
this._setThumbsTranslate(thumbsTranslate - scroll);
} else if (previousIndex > currentIndex) {
this._setThumbsTranslate(thumbsTranslate + scroll);
}
}
}
}
_setThumbsTranslate(thumbsTranslate) {
this.setState({thumbsTranslate});
}
_getThumbsTranslate(indexDifference) {
if (this.props.disableThumbnailScroll) {
return 0;
}
const {thumbnailsWrapperWidth, thumbnailsWrapperHeight} = this.state;
let totalScroll;
if (this._thumbnails) {
// total scroll required to see the last thumbnail
if (this._isThumbnailVertical()) {
if (this._thumbnails.scrollHeight <= thumbnailsWrapperHeight) {
return 0;
}
totalScroll = this._thumbnails.scrollHeight - thumbnailsWrapperHeight;
} else {
if (this._thumbnails.scrollWidth <= thumbnailsWrapperWidth || thumbnailsWrapperWidth <= 0) {
return 0;
}
totalScroll = this._thumbnails.scrollWidth - thumbnailsWrapperWidth;
}
let totalThumbnails = this._thumbnails.children.length;
// scroll-x required per index change
let perIndexScroll = totalScroll / (totalThumbnails - 1);
return indexDifference * perIndexScroll;
}
}
_getAlignmentClassName(index) {
// LEFT, and RIGHT alignments are necessary for lazyLoad
let {currentIndex} = this.state;
let alignment = '';
const LEFT = 'left';
const CENTER = 'center';
const RIGHT = 'right';
switch (index) {
case (currentIndex - 1):
alignment = ` ${LEFT}`;
break;
case (currentIndex):
alignment = ` ${CENTER}`;
break;
case (currentIndex + 1):
alignment = ` ${RIGHT}`;
break;
}
if (this.props.items.length >= 3 && this.props.infinite) {
if (index === 0 && currentIndex === this.props.items.length - 1) {
// set first slide as right slide if were sliding right from last slide
alignment = ` ${RIGHT}`;
} else if (index === this.props.items.length - 1 && currentIndex === 0) {
// set last slide as left slide if were sliding left from first slide
alignment = ` ${LEFT}`;
}
}
return alignment;
}
_isGoingFromFirstToLast() {
const {currentIndex, previousIndex} = this.state;
const totalSlides = this.props.items.length - 1;
return previousIndex === 0 && currentIndex === totalSlides;
}
_isGoingFromLastToFirst() {
const {currentIndex, previousIndex} = this.state;
const totalSlides = this.props.items.length - 1;
return previousIndex === totalSlides && currentIndex === 0;
}
_getTranslateXForTwoSlide(index) {
// For taking care of infinite swipe when there are only two slides
const {currentIndex, offsetPercentage, previousIndex} = this.state;
const baseTranslateX = -100 * currentIndex;
let translateX = baseTranslateX + (index * 100) + offsetPercentage;
// keep track of user swiping direction
if (offsetPercentage > 0) {
this.direction = 'left';
} else if (offsetPercentage < 0) {
this.direction = 'right';
}
// when swiping make sure the slides are on the correct side
if (currentIndex === 0 && index === 1 && offsetPercentage > 0) {
translateX = -100 + offsetPercentage;
} else if (currentIndex === 1 && index === 0 && offsetPercentage < 0) {
translateX = 100 + offsetPercentage;
}
if (currentIndex !== previousIndex) {
// when swiped move the slide to the correct side
if (previousIndex === 0 && index === 0 &&
offsetPercentage === 0 && this.direction === 'left') {
translateX = 100;
} else if (previousIndex === 1 && index === 1 &&
offsetPercentage === 0 && this.direction === 'right') {
translateX = -100;
}
} else {
// keep the slide on the correct slide even when not a swipe
if (currentIndex === 0 && index === 1 &&
offsetPercentage === 0 && this.direction === 'left') {
translateX = -100;
} else if (currentIndex === 1 && index === 0 &&
offsetPercentage === 0 && this.direction === 'right') {
translateX = 100;
}
}
return translateX;
}
_getThumbnailBarHeight() {
if (this._isThumbnailVertical()) {
return {
height: this.state.gallerySlideWrapperHeight
};
}
return {};
}
_shouldPushSlideOnInfiniteMode(index) {
/*
Push(show) slide if slide is the current slide, and the next slide
OR
The slide is going more than 1 slide left, or right, but not going from
first to last and not going from last to first
There is an edge case where if you go to the first or last slide, when they're
not left, or right of each other they will try to catch up in the background
so unless were going from first to last or vice versa we don't want the first
or last slide to show up during our transition
*/
return !this._slideIsTransitioning(index) ||
(this._ignoreIsTransitioning() && !this._isFirstOrLastSlide(index));
}
_slideIsTransitioning(index) {
/*
returns true if the gallery is transitioning and the index is not the
previous or currentIndex
*/
const { isTransitioning, previousIndex, currentIndex } = this.state;
const indexIsNotPreviousOrNextSlide = !(index === previousIndex || index === currentIndex);
return isTransitioning && indexIsNotPreviousOrNextSlide;
}
_isFirstOrLastSlide(index) {
const totalSlides = this.props.items.length - 1;
const isLastSlide = index === totalSlides;
const isFirstSlide = index === 0;
return isLastSlide || isFirstSlide;
}
_ignoreIsTransitioning() {
/*
Ignore isTransitioning because were not going to sibling slides
e.g. center to left or center to right
*/
const { previousIndex, currentIndex } = this.state;
const totalSlides = this.props.items.length - 1;
// we want to show the in between slides transition
const slidingMoreThanOneSlideLeftOrRight = Math.abs(previousIndex - currentIndex) > 1;
const notGoingFromFirstToLast = !(previousIndex === 0 && currentIndex === totalSlides);
const notGoingFromLastToFirst = !(previousIndex === totalSlides && currentIndex === 0);
return slidingMoreThanOneSlideLeftOrRight &&
notGoingFromFirstToLast &&
notGoingFromLastToFirst;
}
_getSlideStyle(index) {
const { currentIndex, offsetPercentage } = this.state;
const { infinite, items, useTranslate3D, isRTL } = this.props;
const baseTranslateX = -100 * currentIndex;
const totalSlides = items.length - 1;
// calculates where the other slides belong based on currentIndex
// if it is RTL the base line should be reversed
let translateX = (baseTranslateX + (index * 100)) * (isRTL ? -1 : 1) + offsetPercentage;
if (infinite && items.length > 2) {
if (currentIndex === 0 && index === totalSlides) {
// make the last slide the slide before the first
// if it is RTL the base line should be reversed
translateX = -100 * (isRTL ? -1 : 1) + offsetPercentage;
} else if (currentIndex === totalSlides && index === 0) {
// make the first slide the slide after the last
// if it is RTL the base line should be reversed
translateX = 100 * (isRTL ? -1 : 1) + offsetPercentage;
}
}
// Special case when there are only 2 items with infinite on
if (infinite && items.length === 2) {
translateX = this._getTranslateXForTwoSlide(index);
}
let translate = `translate(${translateX}%, 0)`;
if (useTranslate3D) {
translate = `translate3d(${translateX}%, 0, 0)`;
}
return {
WebkitTransform: translate,
MozTransform: translate,
msTransform: translate,
OTransform: translate,
transform: translate,
};
}
_getThumbnailStyle() {
let translate;
const { useTranslate3D, isRTL } = this.props;
const { thumbsTranslate } = this.state;
const verticalTranslateValue = isRTL ? thumbsTranslate * -1 : thumbsTranslate;
if (this._isThumbnailVertical()) {
translate = `translate(0, ${thumbsTranslate}px)`;
if (useTranslate3D) {
translate = `translate3d(0, ${thumbsTranslate}px, 0)`;
}
} else {
translate = `translate(${verticalTranslateValue}px, 0)`;
if (useTranslate3D) {
translate = `translate3d(${verticalTranslateValue}px, 0, 0)`;
}
}
return {
WebkitTransform: translate,
MozTransform: translate,
msTransform: translate,
OTransform: translate,
transform: translate
};
}
_slideLeft = () => {
this.props.isRTL ? this._slideNext() : this._slidePrevious();
};
_slideRight = () => {
this.props.isRTL ? this._slidePrevious() : this._slideNext();
};
_slidePrevious = (event) => {
this.slideToIndex(this.state.currentIndex - 1, event);
};
_slideNext = (event) => {
this.slideToIndex(this.state.currentIndex + 1, event);
};
_renderItem = (item) => {
const onImageError = this.props.onImageError || this._handleImageError;
return (
<div className='image-gallery-image'>
{
item.imageSet ?
<picture
onLoad={this.props.onImageLoad}
onError={onImageError}
>
{
item.imageSet.map((source, index) => (
<source
key={index}
media={source.media}
srcSet={source.srcSet}
type={source.type}
/>
))
}
<img
alt={item.originalAlt}
src={item.original}
/>
</picture>
:
<img
src={item.original}
alt={item.originalAlt}
srcSet={item.srcSet}
sizes={item.sizes}
title={item.originalTitle}
onLoad={this.props.onImageLoad}
onError={onImageError}
/>
}
{
item.description &&
<span className='image-gallery-description'>
{item.description}
</span>
}
</div>
);
};
_renderThumbInner = (item) => {
let onThumbnailError = this.props.onThumbnailError || this._handleImageError;
return (
<div className='image-gallery-thumbnail-inner'>
<img
src={item.thumbnail}
alt={item.thumbnailAlt}
title={item.thumbnailTitle}
onError={onThumbnailError}
/>
{item.thumbnailLabel &&
<div className='image-gallery-thumbnail-label'>
{item.thumbnailLabel}
</div>
}
</div>
);
};
_onThumbnailClick = (event, index) => {
this.slideToIndex(index, event);
if (this.props.onThumbnailClick) {
this.props.onThumbnailClick(event, index);
}
};
_onThumbnailMouseOver = (event, index) => {
if (this._thumbnailMouseOverTimer) {
window.clearTimeout(this._thumbnailMouseOverTimer);
this._thumbnailMouseOverTimer = null;
}
this._thumbnailMouseOverTimer = window.setTimeout(() => {
this.slideToIndex(index);
this.pause();
}, 300);
};
_onThumbnailMouseLeave = () => {
if (this._thumbnailMouseOverTimer) {
window.clearTimeout(this._thumbnailMouseOverTimer);
this._thumbnailMouseOverTimer = null;
if (this.props.autoPlay) {
this.play();
}
}
};
render() {
const {
currentIndex,
isFullscreen,
modalFullscreen,
isPlaying,
scrollingLeftRight,
} = this.state;
const {
infinite,
preventDefaultTouchmoveEvent,
slideOnThumbnailOver,
isRTL,
} = this.props;
const thumbnailStyle = this._getThumbnailStyle();
const thumbnailPosition = this.props.thumbnailPosition;
const slideLeft = this._slideLeft;
const slideRight = this._slideRight;
let slides = [];
let thumbnails = [];
let bullets = [];
this.props.items.forEach((item, index) => {
const alignment = this._getAlignmentClassName(index);
const originalClass = item.originalClass ?
` ${item.originalClass}` : '';
const thumbnailClass = item.thumbnailClass ?
` ${item.thumbnailClass}` : '';
const renderItem = item.renderItem ||
this.props.renderItem || this._renderItem;
const renderThumbInner = item.renderThumbInner ||
this.props.renderThumbInner || this._renderThumbInner;
const showItem = !this.props.lazyLoad || alignment || this._lazyLoaded[index];
if (showItem && this.props.lazyLoad) {
this._lazyLoaded[index] = true;
}
let slideStyle = this._getSlideStyle(index);
const slide = (
<div
key={index}
className={'image-gallery-slide' + alignment + originalClass}
style={Object.assign(slideStyle, this.state.style)}
onClick={this.props.onClick}
onTouchMove={this.props.onTouchMove}
onTouchEnd={this.props.onTouchEnd}
onTouchStart={this.props.onTouchStart}
onMouseOver={this.props.onMouseOver}
onMouseLeave={this.props.onMouseLeave}
role={this.props.onClick && 'button'}
>
{showItem ? renderItem(item) : <div style={{ height: '100%' }}></div>}
</div>
);
if (infinite) {
// don't add some slides while transitioning to avoid background transitions
if (this._shouldPushSlideOnInfiniteMode(index)) {
slides.push(slide);
}
} else {
slides.push(slide);
}
if (this.props.showThumbnails) {
thumbnails.push(
<a
key={index}
role='button'
aria-pressed={currentIndex === index ? 'true' : 'false'}
aria-label={`Go to Slide ${index + 1}`}
className={
'image-gallery-thumbnail' +
(currentIndex === index ? ' active' : '') +
thumbnailClass
}
onMouseLeave={slideOnThumbnailOver ? this._onThumbnailMouseLeave : undefined}
onMouseOver={event => slideOnThumbnailOver ? this._onThumbnailMouseOver(event, index) : undefined}
onClick={event => this._onThumbnailClick(event, index)}
>
{renderThumbInner(item)}
</a>
);
}
if (this.props.showBullets) {
const bulletOnClick = event => {
if(item.bulletOnClick){
item.bulletOnClick({item, itemIndex: index, currentIndex});
}
return this.slideToIndex.call(this, index, event);
};
bullets.push(
<button
key={index}
type='button'
className={[
'image-gallery-bullet',
currentIndex === index ? 'active' : '',
item.bulletClass || ''
].join(' ')}
onClick={bulletOnClick}
aria-pressed={currentIndex === index ? 'true' : 'false'}
aria-label={`Go to Slide ${index + 1}`}
>
</button>
);
}
});
const slideWrapper = (
<div
ref={this._initGalleryResizing}
className={`image-gallery-slide-wrapper ${thumbnailPosition} ${isRTL ? 'image-gallery-rtl' : ''}`}
>
{this.props.renderCustomControls && this.props.renderCustomControls()}
{
this.props.showFullscreenButton &&
this.props.renderFullscreenButton(this._toggleFullScreen, isFullscreen)
}
{
this.props.showPlayButton &&
this.props.renderPlayPauseButton(this._togglePlay, isPlaying)
}
{
this._canNavigate() ?
[
this.props.showNav &&
<span key='navigation'>
{this.props.renderLeftNav(slideLeft, !this._canSlideLeft())}
{this.props.renderRightNav(slideRight, !this._canSlideRight())}
</span>,
<Swipeable
className='image-gallery-swipe'
disabled={this.props.disableSwipe}
key='swipeable'
delta={0}
flickThreshold={this.props.flickThreshold}
onSwiping={this._handleSwiping}
onSwiped={this._handleOnSwiped}
stopPropagation={this.props.stopPropagation}
preventDefaultTouchmoveEvent={preventDefaultTouchmoveEvent || scrollingLeftRight}
>
<div className='image-gallery-slides'>
{slides}
</div>
</Swipeable>
]
:
<div className='image-gallery-slides'>
{slides}
</div>
}
{
this.props.showBullets &&
<div className='image-gallery-bullets'>
<div
className='image-gallery-bullets-container'
role='navigation'
aria-label='Bullet Navigation'
>
{bullets}
</div>
</div>
}
{
this.props.showIndex &&
<div className='image-gallery-index'>
<span className='image-gallery-index-current'>
{this.state.currentIndex + 1}
</span>
<span className='image-gallery-index-separator'>
{this.props.indexSeparator}
</span>
<span className='image-gallery-index-total'>
{this.props.items.length}
</span>
</div>
}
</div>
);
const classNames = [
'image-gallery',
this.props.additionalClass,
modalFullscreen ? 'fullscreen-modal' : '',
].filter(name => typeof name === 'string').join(' ');
return (
<div
ref={i => this._imageGallery = i}
className={classNames}
aria-live='polite'
>
<div
className={`image-gallery-content${isFullscreen ? ' fullscreen' : ''}`}
>
{
(thumbnailPosition === 'bottom' || thumbnailPosition === 'right') &&
slideWrapper
}
{
this.props.showThumbnails &&
<div
className={`image-gallery-thumbnails-wrapper ${thumbnailPosition} ${!this._isThumbnailVertical() && isRTL ? 'thumbnails-wrapper-rtl' : ''}`}
style={this._getThumbnailBarHeight()}
>
<div
className='image-gallery-thumbnails'
ref={i => this._thumbnailsWrapper = i}
>
<div
ref={t => this._thumbnails = t}
className='image-gallery-thumbnails-container'
style={thumbnailStyle}
aria-label='Thumbnail Navigation'
>
{thumbnails}
</div>
</div>
</div>
}
{
(thumbnailPosition === 'top' || thumbnailPosition === 'left') &&
slideWrapper
}
</div>
</div>
);
}
}