mk-react-images
Version:
A simple, responsive lightbox component for displaying an array of images with React.js
436 lines (365 loc) • 13.4 kB
JavaScript
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
var _propTypes = require('prop-types');
var _propTypes2 = _interopRequireDefault(_propTypes);
var _react = require('react');
var _react2 = _interopRequireDefault(_react);
var _aphrodite = require('aphrodite');
var _reactScrolllock = require('react-scrolllock');
var _reactScrolllock2 = _interopRequireDefault(_reactScrolllock);
var _theme = require('./theme');
var _theme2 = _interopRequireDefault(_theme);
var _Arrow = require('./components/Arrow');
var _Arrow2 = _interopRequireDefault(_Arrow);
var _Container = require('./components/Container');
var _Container2 = _interopRequireDefault(_Container);
var _Footer = require('./components/Footer');
var _Footer2 = _interopRequireDefault(_Footer);
var _Header = require('./components/Header');
var _Header2 = _interopRequireDefault(_Header);
var _PaginatedThumbnails = require('./components/PaginatedThumbnails');
var _PaginatedThumbnails2 = _interopRequireDefault(_PaginatedThumbnails);
var _Portal = require('./components/Portal');
var _Portal2 = _interopRequireDefault(_Portal);
var _bindFunctions = require('./utils/bindFunctions');
var _bindFunctions2 = _interopRequireDefault(_bindFunctions);
var _canUseDom = require('./utils/canUseDom');
var _canUseDom2 = _interopRequireDefault(_canUseDom);
var _deepMerge = require('./utils/deepMerge');
var _deepMerge2 = _interopRequireDefault(_deepMerge);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
var Lightbox = function (_Component) {
_inherits(Lightbox, _Component);
function Lightbox(props) {
_classCallCheck(this, Lightbox);
var _this = _possibleConstructorReturn(this, (Lightbox.__proto__ || Object.getPrototypeOf(Lightbox)).call(this, props));
_this.theme = (0, _deepMerge2.default)(_theme2.default, props.theme);
_this.classes = _aphrodite.StyleSheet.create((0, _deepMerge2.default)(defaultStyles, _this.theme));
_bindFunctions2.default.call(_this, ['gotoNext', 'gotoPrev', 'closeBackdrop', 'handleKeyboardInput']);
return _this;
}
_createClass(Lightbox, [{
key: 'getChildContext',
value: function getChildContext() {
return {
theme: this.theme
};
}
}, {
key: 'componentDidMount',
value: function componentDidMount() {
if (this.props.isOpen && this.props.enableKeyboardInput) {
window.addEventListener('keydown', this.handleKeyboardInput);
}
}
}, {
key: 'componentWillReceiveProps',
value: function componentWillReceiveProps(nextProps) {
if (!_canUseDom2.default) return;
// preload images
if (nextProps.preloadNextImage) {
var currentIndex = this.props.currentImage;
var nextIndex = nextProps.currentImage + 1;
var prevIndex = nextProps.currentImage - 1;
var preloadIndex = void 0;
if (currentIndex && nextProps.currentImage > currentIndex) {
preloadIndex = nextIndex;
} else if (currentIndex && nextProps.currentImage < currentIndex) {
preloadIndex = prevIndex;
}
// if we know the user's direction just get one image
// otherwise, to be safe, we need to grab one in each direction
if (preloadIndex) {
this.preloadImage(preloadIndex);
} else {
this.preloadImage(prevIndex);
this.preloadImage(nextIndex);
}
}
// add/remove event listeners
if (!this.props.isOpen && nextProps.isOpen && nextProps.enableKeyboardInput) {
window.addEventListener('keydown', this.handleKeyboardInput);
}
if (!nextProps.isOpen && nextProps.enableKeyboardInput) {
window.removeEventListener('keydown', this.handleKeyboardInput);
}
}
}, {
key: 'componentWillUnmount',
value: function componentWillUnmount() {
if (this.props.enableKeyboardInput) {
window.removeEventListener('keydown', this.handleKeyboardInput);
}
}
// ==============================
// METHODS
// ==============================
}, {
key: 'preloadImage',
value: function preloadImage(idx) {
var image = this.props.images[idx];
if (!image) return;
var img = new Image();
img.src = image.src;
img.srcset = img.srcSet || img.srcset;
if (image.srcset) {
img.srcset = image.srcset.join();
}
}
}, {
key: 'gotoNext',
value: function gotoNext(event) {
if (this.props.currentImage === this.props.images.length - 1) return;
if (event) {
event.preventDefault();
event.stopPropagation();
}
this.props.onClickNext();
}
}, {
key: 'gotoPrev',
value: function gotoPrev(event) {
if (this.props.currentImage === 0) return;
if (event) {
event.preventDefault();
event.stopPropagation();
}
this.props.onClickPrev();
}
}, {
key: 'closeBackdrop',
value: function closeBackdrop(event) {
// make sure event only happens if they click the backdrop
// and if the caption is widening the figure element let that respond too
if (event.target.id === 'lightboxBackdrop' || event.target.tagName === 'FIGURE') {
this.props.onClose();
}
}
}, {
key: 'handleKeyboardInput',
value: function handleKeyboardInput(event) {
if (event.keyCode === 37) {
// left
this.gotoPrev(event);
return true;
} else if (event.keyCode === 39) {
// right
this.gotoNext(event);
return true;
} else if (event.keyCode === 27) {
// esc
this.props.onClose();
return true;
}
return false;
}
// ==============================
// RENDERERS
// ==============================
}, {
key: 'renderArrowPrev',
value: function renderArrowPrev() {
if (this.props.currentImage === 0) return null;
return _react2.default.createElement(_Arrow2.default, {
direction: 'left',
icon: 'arrowLeft',
onClick: this.gotoPrev,
title: this.props.leftArrowTitle,
type: 'button'
});
}
}, {
key: 'renderArrowNext',
value: function renderArrowNext() {
if (this.props.currentImage === this.props.images.length - 1) return null;
return _react2.default.createElement(_Arrow2.default, {
direction: 'right',
icon: 'arrowRight',
onClick: this.gotoNext,
title: this.props.rightArrowTitle,
type: 'button'
});
}
}, {
key: 'renderDialog',
value: function renderDialog() {
var _props = this.props,
backdropClosesModal = _props.backdropClosesModal,
customControls = _props.customControls,
isOpen = _props.isOpen,
onClose = _props.onClose,
showCloseButton = _props.showCloseButton,
showThumbnails = _props.showThumbnails,
width = _props.width;
if (!isOpen) return _react2.default.createElement('span', { key: 'closed' });
var offsetThumbnails = 0;
if (showThumbnails) {
offsetThumbnails = this.theme.thumbnail.size + this.theme.container.gutter.vertical;
}
return _react2.default.createElement(
_Container2.default,
{
key: 'open',
onClick: backdropClosesModal && this.closeBackdrop,
onTouchEnd: backdropClosesModal && this.closeBackdrop
},
_react2.default.createElement(
'div',
{ className: (0, _aphrodite.css)(this.classes.content), style: { marginBottom: offsetThumbnails, maxWidth: width } },
_react2.default.createElement(_Header2.default, {
customControls: customControls,
onClose: onClose,
showCloseButton: showCloseButton,
closeButtonTitle: this.props.closeButtonTitle
}),
this.renderImages()
),
this.renderThumbnails(),
this.renderArrowPrev(),
this.renderArrowNext()
);
}
}, {
key: 'renderImages',
value: function renderImages() {
var _props2 = this.props,
currentImage = _props2.currentImage,
images = _props2.images,
imageCountSeparator = _props2.imageCountSeparator,
onClickImage = _props2.onClickImage,
showImageCount = _props2.showImageCount,
showThumbnails = _props2.showThumbnails;
if (!images || !images.length) return null;
var image = images[currentImage];
image.srcset = image.srcSet || image.srcset;
var srcset = void 0;
var sizes = void 0;
if (image.srcset) {
srcset = image.srcset.join();
sizes = '100vw';
}
var thumbnailsSize = showThumbnails ? this.theme.thumbnail.size : 0;
var heightOffset = this.theme.header.height + this.theme.footer.height + thumbnailsSize + this.theme.container.gutter.vertical + 'px';
return _react2.default.createElement(
'figure',
{ className: (0, _aphrodite.css)(this.classes.figure) },
_react2.default.createElement('img', {
className: (0, _aphrodite.css)(this.classes.image),
onClick: !!onClickImage && onClickImage,
sizes: sizes,
alt: image.alt,
src: image.src,
srcSet: srcset,
style: {
cursor: this.props.onClickImage ? 'pointer' : 'auto',
maxHeight: 'calc(100vh - ' + heightOffset + ')'
}
}),
_react2.default.createElement(_Footer2.default, {
caption: images[currentImage].caption,
countCurrent: currentImage + 1,
countSeparator: imageCountSeparator,
countTotal: images.length,
showCount: showImageCount
})
);
}
}, {
key: 'renderThumbnails',
value: function renderThumbnails() {
var _props3 = this.props,
images = _props3.images,
currentImage = _props3.currentImage,
onClickThumbnail = _props3.onClickThumbnail,
showThumbnails = _props3.showThumbnails,
thumbnailOffset = _props3.thumbnailOffset;
if (!showThumbnails) return;
return _react2.default.createElement(_PaginatedThumbnails2.default, {
currentImage: currentImage,
images: images,
offset: thumbnailOffset,
onClickThumbnail: onClickThumbnail
});
}
}, {
key: 'render',
value: function render() {
return _react2.default.createElement(
_Portal2.default,
null,
this.renderDialog()
);
}
}]);
return Lightbox;
}(_react.Component);
Lightbox.propTypes = {
backdropClosesModal: _propTypes2.default.bool,
closeButtonTitle: _propTypes2.default.string,
currentImage: _propTypes2.default.number,
customControls: _propTypes2.default.arrayOf(_propTypes2.default.node),
enableKeyboardInput: _propTypes2.default.bool,
imageCountSeparator: _propTypes2.default.string,
images: _propTypes2.default.arrayOf(_propTypes2.default.shape({
src: _propTypes2.default.string.isRequired,
srcset: _propTypes2.default.array,
caption: _propTypes2.default.oneOfType([_propTypes2.default.string, _propTypes2.default.element]),
thumbnail: _propTypes2.default.string
})).isRequired,
isOpen: _propTypes2.default.bool,
leftArrowTitle: _propTypes2.default.string,
onClickImage: _propTypes2.default.func,
onClickNext: _propTypes2.default.func,
onClickPrev: _propTypes2.default.func,
onClose: _propTypes2.default.func.isRequired,
preloadNextImage: _propTypes2.default.bool,
rightArrowTitle: _propTypes2.default.string,
showCloseButton: _propTypes2.default.bool,
showImageCount: _propTypes2.default.bool,
showThumbnails: _propTypes2.default.bool,
theme: _propTypes2.default.object,
thumbnailOffset: _propTypes2.default.number,
width: _propTypes2.default.number
};
Lightbox.defaultProps = {
closeButtonTitle: 'Close (Esc)',
currentImage: 0,
enableKeyboardInput: true,
imageCountSeparator: ' of ',
leftArrowTitle: 'Previous (Left arrow key)',
onClickShowNextImage: true,
preloadNextImage: true,
rightArrowTitle: 'Next (Right arrow key)',
showCloseButton: true,
showImageCount: true,
theme: {},
thumbnailOffset: 2,
width: 1024
};
Lightbox.childContextTypes = {
theme: _propTypes2.default.object.isRequired
};
var defaultStyles = {
content: {
position: 'relative'
},
figure: {
margin: 0 // remove browser default
},
image: {
display: 'block', // removes browser default gutter
height: 'auto',
margin: '0 auto', // maintain center on very short screens OR very narrow image
maxWidth: '100%',
// disable user select
WebkitTouchCallout: 'none',
userSelect: 'none'
}
};
exports.default = Lightbox;