react-image-hotspots
Version:
React component for rendering images with hotspots
466 lines (465 loc) • 67.2 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
Object.defineProperty(exports, "default", {
enumerable: true,
get: function() {
return _default;
}
});
const _react = /*#__PURE__*/ _interop_require_default(require("react"));
const _proptypes = /*#__PURE__*/ _interop_require_default(require("prop-types"));
const _Hotspot = /*#__PURE__*/ _interop_require_default(require("./Hotspot"));
function _define_property(obj, key, value) {
if (key in obj) {
Object.defineProperty(obj, key, {
value: value,
enumerable: true,
configurable: true,
writable: true
});
} else {
obj[key] = value;
}
return obj;
}
function _interop_require_default(obj) {
return obj && obj.__esModule ? obj : {
default: obj
};
}
class ImageHotspots extends _react.default.Component {
constructor(props){
super(props), _define_property(this, "componentDidMount", ()=>{
const { hideFullscreenControl, hideZoomControls, hideMinimap, hotspots, background } = this.props;
const { offsetWidth: width, offsetHeight: height } = this.container.current;
const orientation = width > height ? 'landscape' : 'portrait';
const ratio = orientation === 'landscape' ? width / height : height / width;
this.setState({
container: {
width,
height,
ratio,
orientation,
background
},
hideFullscreenControl,
hideZoomControls,
hideMinimap,
hotspots
});
window.addEventListener('resize', this.onWindowResize);
}), _define_property(this, "componentWillUnmount", ()=>{
window.removeEventListener('resize', this.onWindowResize);
}), _define_property(this, "startDrag", (event, element)=>{
const cursorX = event.clientX;
const cursorY = event.clientY;
if (element === 'image') {
this.setState((state)=>({
...state,
cursorX,
cursorY,
dragging: true
}));
} else if (element === 'guide') {
// TODO
}
event.preventDefault();
}), _define_property(this, "whileDrag", (event)=>{
const { image, minimap } = this.state;
const cursorX = event.clientX;
const cursorY = event.clientY;
const deltaX = cursorX - this.state.cursorX;
const deltaY = cursorY - this.state.cursorY;
const newOffsetX = image.offsetX + deltaX;
const newOffsetY = image.offsetY + deltaY;
this.setState((state)=>({
...state,
cursorX,
cursorY,
image: {
...image,
offsetX: newOffsetX,
offsetY: newOffsetY
},
minimap: {
...minimap,
offsetX: -(minimap.width / image.width * newOffsetX),
offsetY: -(minimap.height / image.height * newOffsetY)
}
}));
}), _define_property(this, "stopDrag", ()=>{
const { container, image, minimap } = this.state;
const deltaX = container.width - image.width - image.offsetX;
const deltaY = container.height - image.height - image.offsetY;
const offsetXMax = container.orientation === image.orientation ? -Math.abs(image.width - container.width) : -Math.abs(container.width - image.width);
const offsetYMax = container.orientation === image.orientation ? -Math.abs(container.height - image.height) : -Math.abs(image.height - container.height);
this.setState((state)=>({
...state,
image: {
...state.image,
offsetX: image.offsetX >= 0 ? 0 : deltaX >= 0 ? offsetXMax : image.offsetX,
offsetY: image.offsetY >= 0 ? container.height > image.height ? container.height / 2 - image.height / 2 : 0 : deltaY >= 0 ? container.height > image.height ? container.height / 2 - image.height / 2 : offsetYMax : image.offsetY
},
minimap: {
...state.minimap,
offsetX: image.offsetX >= 0 || image.width < container.width ? 0 : deltaX >= 0 ? -(minimap.height / image.height * offsetXMax) : -(minimap.height / image.height * image.offsetX),
offsetY: image.offsetY >= 0 || image.height < container.height ? 0 : deltaY >= 0 ? -(minimap.height / image.height * offsetYMax) : -(minimap.height / image.height * image.offsetY)
},
dragging: false
}));
}), _define_property(this, "handleOnImageLoad", ({ target: image })=>{
const { offsetWidth: initialWidth, offsetHeight: initialHeight } = image;
const { container, minimap, hideZoomControls, hideMinimap } = this.state;
const orientation = initialWidth > initialHeight ? 'landscape' : 'portrait';
const ratio = orientation === 'landscape' ? initialWidth / initialHeight : initialHeight / initialWidth;
const width = container.orientation === orientation ? orientation === 'landscape' ? ratio >= container.ratio ? container.width // landscape image bigger than landscape container
: container.height * ratio // landscape image smaller than landscape container
: ratio >= container.ratio ? container.height / ratio // portrait image bigger than portrait container
: container.width // portrait image smaller than portrait container
: orientation === 'landscape' ? container.width // landscape image and portrait container
: container.height / ratio // portrait image and landscape container
;
const height = container.orientation === orientation ? orientation === 'landscape' ? ratio >= container.ratio ? container.width / ratio // landscape image bigger than landscape container
: container.height // landscape image smaller than landscape container
: ratio >= container.ratio ? container.height // portrait image bigger than portrait container
: container.width * ratio // portrait image smaller than portrait container
: orientation === 'landscape' ? container.width / ratio // landscape image and portrait container
: container.height // portrait image and landscape container
;
const resizable = initialWidth > width || initialHeight > height;
this.setState((prevState)=>({
...prevState,
image: {
...prevState.image,
initialWidth,
initialHeight,
width,
height,
scale: 1,
ratio,
orientation,
offsetX: 0,
offsetY: container.height / 2 - height / 2
},
minimap: {
...minimap,
width: orientation === 'landscape' ? minimap.initialSize : minimap.initialSize / ratio,
height: orientation === 'portrait' ? minimap.initialSize : minimap.initialSize / ratio,
guideWidth: orientation === 'landscape' ? minimap.initialSize : minimap.initialSize / ratio,
guideHeight: orientation === 'portrait' ? minimap.initialSize : minimap.initialSize / ratio
},
hideZoomControls: hideZoomControls || !resizable,
hideMinimap: hideMinimap || !resizable,
resizable,
draggable: false
}));
}), _define_property(this, "onWindowResize", ()=>{
const { offsetWidth: width, offsetHeight: height } = this.container.current;
const orientation = width > height ? 'landscape' : 'portrait';
const ratio = orientation === 'landscape' ? width / height : height / width;
this.setState({
container: {
width,
height,
ratio,
orientation
}
});
this.zoom(this.state.image.scale);
}), _define_property(this, "toggleFullscreen", ()=>{
const { fullscreen } = this.state;
if (!fullscreen) {
this.requestFullscreen(this.container.current);
this.setState({
fullscreen: true
});
} else {
this.exitFullscreen();
this.setState({
fullscreen: false
});
}
}), _define_property(this, "zoom", (scale)=>{
if (scale > 0) {
const { container, image, minimap } = this.state;
const width = container.orientation === image.orientation ? image.orientation === 'landscape' ? image.ratio >= container.ratio ? container.width * scale // landscape image bigger than landscape container
: container.height * image.ratio * scale // landscape image smaller than landscape container
: image.ratio >= container.ratio ? container.height / image.ratio * scale // portrait image bigger than portrait container
: container.width * scale // portrait image smaller than portrait container
: image.orientation === 'landscape' ? container.width * scale // landscape image and portrait container
: container.height / image.ratio * scale // portrait image and landscape container
;
const height = container.orientation === image.orientation ? image.orientation === 'landscape' ? image.ratio >= container.ratio ? container.width / image.ratio * scale // landscape image bigger than landscape container
: container.height * scale // landscape image smaller than landscape container
: image.ratio >= container.ratio ? container.height * scale // portrait image bigger than portrait container
: container.width * image.ratio * scale // portrait image smaller than portrait container
: image.orientation === 'landscape' ? container.width / image.ratio * scale // landscape image and portrait container
: container.height * scale // portrait image and landscape container
;
const guideWidth = container.width >= width ? minimap.width : minimap.width / (width / container.width);
const guideHeight = container.height >= height ? minimap.height : minimap.height / (height / container.height);
const deltaX = Math.round(width - image.width);
const deltaY = Math.round(height - image.height);
const guideDeltaX = Math.round(guideWidth - minimap.guideWidth);
const guideDeltaY = Math.round(guideHeight - minimap.guideHeight);
const offsetX = image.offsetX - deltaX / 2;
const offsetY = image.offsetY - deltaY / 2;
const guideOffsetX = Math.round(minimap.offsetX - guideDeltaX / 2);
const guideOffsetY = Math.round(minimap.offsetY - guideDeltaY / 2);
const offsetXMax = -Math.abs(Math.round(container.width - width));
const offsetYMax = -Math.abs(Math.round(container.height - height));
const guideOffsetXMax = Math.round(minimap.width - guideWidth);
const guideOffsetYMax = Math.round(minimap.height - guideHeight);
if (image.initialWidth > width && image.initialHeight > height) {
this.setState((prevState)=>({
image: {
...prevState.image,
width,
height,
scale,
offsetX: offsetX >= 0 || container.width > width ? 0 : image.offsetX <= offsetXMax ? offsetXMax : offsetX,
offsetY: container.height > height ? container.height / 2 - height / 2 : offsetY >= 0 ? 0 : image.offsetY < offsetYMax ? offsetYMax : offsetY
},
minimap: {
...prevState.minimap,
guideWidth,
guideHeight,
offsetX: guideOffsetX <= 0 ? 0 : minimap.offsetX < guideOffsetXMax ? guideOffsetX : guideOffsetXMax,
offsetY: guideOffsetY <= 0 || height < container.height ? 0 : minimap.offsetY < guideOffsetYMax ? guideOffsetY : guideOffsetYMax
},
draggable: scale > 1
}));
}
// Reset image position
if (scale === 1) {
this.setState((prevState)=>({
image: {
...prevState.image,
offsetX: 0,
offsetY: container.height / 2 - height / 2
},
minimap: {
...prevState.minimap,
offsetX: 0,
offsetY: 0
}
}));
}
}
}), _define_property(this, "requestFullscreen", (element)=>{
if (element.requestFullscreen) {
element.requestFullscreen();
} else if (element.mozRequestFullScreen) {
element.mozRequestFullScreen();
} else if (element.webkitRequestFullscreen) {
element.webkitRequestFullscreen();
} else if (element.msRequestFullscreen) {
element.msRequestFullscreen();
}
}), _define_property(this, "exitFullscreen", ()=>{
if (document.exitFullscreen) {
document.exitFullscreen();
} else if (document.mozCancelFullScreen) {
document.mozCancelFullScreen();
} else if (document.webkitExitFullscreen) {
document.webkitExitFullscreen();
} else if (document.msExitFullscreen) {
document.msExitFullscreen();
}
}), _define_property(this, "render", ()=>{
const { src, alt, hotspots, background } = this.props;
const { container, image, minimap, fullscreen, dragging, hideFullscreenControl, hideZoomControls, hideMinimap, draggable } = this.state;
const imageLoaded = image.initialWidth && image.initialHeight;
const containerStyle = {
width: '100%',
height: '100%',
position: 'relative',
overflow: 'hidden',
textAlign: 'center',
background: background || container.background
};
const imageStyle = {
position: 'relative',
left: image.offsetX,
top: image.offsetY
};
const hotspotsStyle = {
position: 'absolute',
top: image.offsetY,
left: image.offsetX,
right: image.offsetX >= 0 ? 0 : 'auto',
margin: 'auto',
pointerEvents: 'none'
};
const topControlsStyle = {
position: 'absolute',
top: 10,
right: 10,
pointerEvents: this.state.dragging ? 'none' : 'auto'
};
const bottomControlsStyle = {
position: 'absolute',
bottom: 10,
right: 10,
pointerEvents: this.state.dragging ? 'none' : 'auto'
};
const buttonStyle = {
width: '25px',
height: '25px',
border: 'none',
background: '#fff',
boxShadow: '0px 0px 2px 0px rgba(0,0,0,0.5)'
};
const minimapStyle = {
width: minimap.width,
height: minimap.height,
position: 'absolute',
display: 'block',
bottom: 10,
left: 10,
background: '#fff',
boxShadow: '0px 0px 2px 0px rgba(0,0,0,0.5)',
pointerEvents: 'none'
};
const guideStyle = {
width: minimap.guideWidth,
height: minimap.guideHeight,
position: 'absolute',
display: 'block',
left: minimap.offsetX,
top: minimap.offsetY,
border: '1px solid rgba(64, 139, 252, 0.8)',
background: 'rgba(64, 139, 252, 0.1)',
pointerEvents: 'none'
};
if (imageLoaded) {
if (container.orientation === 'landscape') {
imageStyle.height = image.height;
} else {
imageStyle.width = image.width;
}
if (image.orientation === 'landscape') {
hotspotsStyle.width = image.width;
hotspotsStyle.height = image.width / image.ratio;
} else {
hotspotsStyle.width = image.height / image.ratio;
hotspotsStyle.height = image.height;
}
}
return /*#__PURE__*/ _react.default.createElement("div", {
ref: this.container,
style: containerStyle,
onMouseOut: (event)=>{
if (dragging) {
this.stopDrag(event);
}
},
onBlur: (event)=>{
if (dragging) {
this.stopDrag(event);
}
}
}, src && /*#__PURE__*/ _react.default.createElement("img", {
src: src,
alt: alt,
onLoad: this.handleOnImageLoad,
style: imageStyle,
onMouseDown: (event)=>{
if (!hideZoomControls && draggable) {
this.startDrag(event, 'image');
}
},
onMouseMove: (event)=>{
if (!hideZoomControls && dragging) {
this.whileDrag(event);
}
},
onMouseUp: (event)=>{
if (dragging) {
this.stopDrag(event);
}
}
}), hotspots && /*#__PURE__*/ _react.default.createElement("div", {
style: hotspotsStyle
}, hotspots.map((hotspot, i)=>/*#__PURE__*/ _react.default.createElement(_Hotspot.default, {
key: i,
...hotspot
}))), !hideFullscreenControl && /*#__PURE__*/ _react.default.createElement("div", {
style: topControlsStyle
}, /*#__PURE__*/ _react.default.createElement("button", {
style: buttonStyle,
onClick: ()=>this.toggleFullscreen()
}, fullscreen ? 'X' : 'FS')), !hideZoomControls && /*#__PURE__*/ _react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/ _react.default.createElement("div", {
style: bottomControlsStyle
}, draggable && /*#__PURE__*/ _react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/ _react.default.createElement("button", {
style: buttonStyle,
onClick: ()=>this.zoom(1)
}, "Fit"), /*#__PURE__*/ _react.default.createElement("br", null), /*#__PURE__*/ _react.default.createElement("br", null)), /*#__PURE__*/ _react.default.createElement("button", {
style: buttonStyle,
onClick: ()=>this.zoom(image.scale + 1)
}, "+"), /*#__PURE__*/ _react.default.createElement("br", null), /*#__PURE__*/ _react.default.createElement("button", {
style: buttonStyle,
onClick: ()=>this.zoom(image.scale - 1)
}, "-")), !hideMinimap && /*#__PURE__*/ _react.default.createElement("div", {
style: minimapStyle
}, src && /*#__PURE__*/ _react.default.createElement("img", {
src: src,
width: minimapStyle.width,
height: minimapStyle.height
}), /*#__PURE__*/ _react.default.createElement("div", {
style: guideStyle
}))));
});
this.state = {
container: {
width: undefined,
height: undefined,
ratio: undefined,
orientation: undefined,
background: undefined
},
image: {
initialWidth: undefined,
initialHeight: undefined,
width: undefined,
height: undefined,
scale: undefined,
ratio: undefined,
orientation: undefined,
offsetX: undefined,
offsetY: undefined
},
minimap: {
initialSize: 100,
width: undefined,
height: undefined,
guideWidth: undefined,
guideHeight: undefined,
offsetX: 0,
offsetY: 0
},
hideFullscreenControl: false,
hideZoomControls: false,
hideMinimap: false,
resizable: undefined,
draggable: undefined,
cursorX: undefined,
cursorY: undefined,
mcursorX: undefined,
mcursorY: undefined,
dragging: undefined,
isGuideDragging: undefined,
hotspots: []
};
this.container = /*#__PURE__*/ _react.default.createRef();
}
}
ImageHotspots.propTypes = {
src: _proptypes.default.string,
alt: _proptypes.default.string,
hotspots: _proptypes.default.array
};
const _default = ImageHotspots;
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/ImageHotspots.js"],"sourcesContent":["import React from 'react'\nimport PropTypes from 'prop-types'\nimport Hotspot from './Hotspot'\n\nclass ImageHotspots extends React.Component {\n  constructor (props) {\n    super(props)\n\n    this.state = {\n      container: {\n        width: undefined,\n        height: undefined,\n        ratio: undefined,\n        orientation: undefined,\n        background: undefined\n      },\n      image: {\n        initialWidth: undefined,\n        initialHeight: undefined,\n        width: undefined,\n        height: undefined,\n        scale: undefined,\n        ratio: undefined,\n        orientation: undefined,\n        offsetX: undefined,\n        offsetY: undefined\n      },\n      minimap: {\n        initialSize: 100,\n        width: undefined,\n        height: undefined,\n        guideWidth: undefined,\n        guideHeight: undefined,\n        offsetX: 0,\n        offsetY: 0\n      },\n      hideFullscreenControl: false,\n      hideZoomControls: false,\n      hideMinimap: false,\n      resizable: undefined,\n      draggable: undefined,\n      cursorX: undefined,\n      cursorY: undefined,\n      mcursorX: undefined,\n      mcursorY: undefined,\n      dragging: undefined,\n      isGuideDragging: undefined,\n      hotspots: []\n    }\n\n    this.container = React.createRef()\n  }\n\n  componentDidMount = () => {\n    const {\n      hideFullscreenControl,\n      hideZoomControls,\n      hideMinimap,\n      hotspots,\n      background\n    } = this.props\n    const { offsetWidth: width, offsetHeight: height } = this.container.current\n    const orientation = (width > height) ? 'landscape' : 'portrait'\n    const ratio = (orientation === 'landscape') ? width / height : height / width\n\n    this.setState({\n      container: { width, height, ratio, orientation, background },\n      hideFullscreenControl,\n      hideZoomControls,\n      hideMinimap,\n      hotspots\n    })\n\n    window.addEventListener('resize', this.onWindowResize)\n  }\n\n  componentWillUnmount = () => {\n    window.removeEventListener('resize', this.onWindowResize)\n  }\n\n  startDrag = (event, element) => {\n    const cursorX = event.clientX\n    const cursorY = event.clientY\n    if (element === 'image') {\n      this.setState(state => ({\n        ...state,\n        cursorX,\n        cursorY,\n        dragging: true\n      }))\n    } else if (element === 'guide') {\n      // TODO\n    }\n    event.preventDefault()\n  }\n\n  whileDrag = (event) => {\n    const { image, minimap } = this.state\n    const cursorX = event.clientX\n    const cursorY = event.clientY\n    const deltaX = cursorX - this.state.cursorX\n    const deltaY = cursorY - this.state.cursorY\n    const newOffsetX = image.offsetX + deltaX\n    const newOffsetY = image.offsetY + deltaY\n\n    this.setState(state => ({\n      ...state,\n      cursorX,\n      cursorY,\n      image: {\n        ...image,\n        offsetX: newOffsetX,\n        offsetY: newOffsetY\n      },\n      minimap: {\n        ...minimap,\n        offsetX: -(minimap.width / image.width * newOffsetX),\n        offsetY: -(minimap.height / image.height * newOffsetY)\n      }\n    }))\n  }\n\n  stopDrag = () => {\n    const { container, image, minimap } = this.state\n    const deltaX = container.width - image.width - image.offsetX\n    const deltaY = container.height - image.height - image.offsetY\n\n    const offsetXMax = container.orientation === image.orientation\n      ? -Math.abs(image.width - container.width)\n      : -Math.abs(container.width - image.width)\n    const offsetYMax = container.orientation === image.orientation\n      ? -Math.abs(container.height - image.height)\n      : -Math.abs(image.height - container.height)\n\n    this.setState(state => ({\n      ...state,\n      image: {\n        ...state.image,\n        offsetX: image.offsetX >= 0\n          ? 0\n          : deltaX >= 0\n            ? offsetXMax\n            : image.offsetX,\n        offsetY: image.offsetY >= 0\n          ? (container.height > image.height)\n              ? container.height / 2 - image.height / 2\n              : 0\n          : deltaY >= 0\n            ? (container.height > image.height)\n                ? container.height / 2 - image.height / 2\n                : offsetYMax\n            : image.offsetY\n      },\n      minimap: {\n        ...state.minimap,\n        offsetX: image.offsetX >= 0 || image.width < container.width\n          ? 0\n          : deltaX >= 0\n            ? -(minimap.height / image.height * offsetXMax)\n            : -(minimap.height / image.height * image.offsetX),\n        offsetY: image.offsetY >= 0 || image.height < container.height\n          ? 0\n          : deltaY >= 0\n            ? -(minimap.height / image.height * offsetYMax)\n            : -(minimap.height / image.height * image.offsetY)\n      },\n      dragging: false\n    }))\n  }\n\n  handleOnImageLoad = ({ target: image }) => {\n    const { offsetWidth: initialWidth, offsetHeight: initialHeight } = image\n    const { container, minimap, hideZoomControls, hideMinimap } = this.state\n    const orientation = (initialWidth > initialHeight) ? 'landscape' : 'portrait'\n    const ratio = (orientation === 'landscape')\n      ? initialWidth / initialHeight\n      : initialHeight / initialWidth\n\n    const width = container.orientation === orientation\n      ? orientation === 'landscape'\n          ? ratio >= container.ratio\n              ? container.width // landscape image bigger than landscape container\n              : container.height * ratio // landscape image smaller than landscape container\n          : ratio >= container.ratio\n            ? container.height / ratio // portrait image bigger than portrait container\n            : container.width // portrait image smaller than portrait container\n      : orientation === 'landscape'\n        ? container.width // landscape image and portrait container\n        : container.height / ratio // portrait image and landscape container\n\n    const height = container.orientation === orientation\n      ? orientation === 'landscape'\n          ? ratio >= container.ratio\n              ? container.width / ratio // landscape image bigger than landscape container\n              : container.height // landscape image smaller than landscape container\n          : ratio >= container.ratio\n            ? container.height // portrait image bigger than portrait container\n            : container.width * ratio // portrait image smaller than portrait container\n      : orientation === 'landscape'\n        ? container.width / ratio // landscape image and portrait container\n        : container.height // portrait image and landscape container\n\n    const resizable = (initialWidth > width) || (initialHeight > height)\n\n    this.setState((prevState) => ({\n      ...prevState,\n      image: {\n        ...prevState.image,\n        initialWidth,\n        initialHeight,\n        width,\n        height,\n        scale: 1,\n        ratio,\n        orientation,\n        offsetX: 0,\n        offsetY: container.height / 2 - height / 2\n      },\n      minimap: {\n        ...minimap,\n        width: orientation === 'landscape'\n          ? minimap.initialSize\n          : minimap.initialSize / ratio,\n        height: orientation === 'portrait'\n          ? minimap.initialSize\n          : minimap.initialSize / ratio,\n        guideWidth: orientation === 'landscape'\n          ? minimap.initialSize\n          : minimap.initialSize / ratio,\n        guideHeight: orientation === 'portrait'\n          ? minimap.initialSize\n          : minimap.initialSize / ratio\n      },\n      hideZoomControls: hideZoomControls || !resizable,\n      hideMinimap: hideMinimap || !resizable,\n      resizable,\n      draggable: false\n    }))\n  }\n\n  onWindowResize = () => {\n    const { offsetWidth: width, offsetHeight: height } = this.container.current\n    const orientation = (width > height) ? 'landscape' : 'portrait'\n    const ratio = (orientation === 'landscape') ? width / height : height / width\n\n    this.setState({ container: { width, height, ratio, orientation } })\n\n    this.zoom(this.state.image.scale)\n  }\n\n  toggleFullscreen = () => {\n    const { fullscreen } = this.state\n    if (!fullscreen) {\n      this.requestFullscreen(this.container.current)\n      this.setState({ fullscreen: true })\n    } else {\n      this.exitFullscreen()\n      this.setState({ fullscreen: false })\n    }\n  }\n\n  zoom = (scale) => {\n    if (scale > 0) {\n      const { container, image, minimap } = this.state\n\n      const width = container.orientation === image.orientation\n        ? image.orientation === 'landscape'\n            ? image.ratio >= container.ratio\n                ? container.width * scale// landscape image bigger than landscape container\n                : container.height * image.ratio * scale// landscape image smaller than landscape container\n            : image.ratio >= container.ratio\n              ? container.height / image.ratio * scale// portrait image bigger than portrait container\n              : container.width * scale// portrait image smaller than portrait container\n        : image.orientation === 'landscape'\n          ? container.width * scale// landscape image and portrait container\n          : container.height / image.ratio * scale// portrait image and landscape container\n\n      const height = container.orientation === image.orientation\n        ? image.orientation === 'landscape'\n            ? image.ratio >= container.ratio\n                ? container.width / image.ratio * scale// landscape image bigger than landscape container\n                : container.height * scale// landscape image smaller than landscape container\n            : image.ratio >= container.ratio\n              ? container.height * scale// portrait image bigger than portrait container\n              : container.width * image.ratio * scale// portrait image smaller than portrait container\n        : image.orientation === 'landscape'\n          ? container.width / image.ratio * scale// landscape image and portrait container\n          : container.height * scale// portrait image and landscape container\n\n      const guideWidth = (container.width >= width)\n        ? minimap.width\n        : minimap.width / (width / container.width)\n      const guideHeight = (container.height >= height)\n        ? minimap.height\n        : minimap.height / (height / container.height)\n\n      const deltaX = Math.round(width - image.width)\n      const deltaY = Math.round(height - image.height)\n      const guideDeltaX = Math.round(guideWidth - minimap.guideWidth)\n      const guideDeltaY = Math.round(guideHeight - minimap.guideHeight)\n\n      const offsetX = image.offsetX - deltaX / 2\n      const offsetY = image.offsetY - deltaY / 2\n      const guideOffsetX = Math.round(minimap.offsetX - guideDeltaX / 2)\n      const guideOffsetY = Math.round(minimap.offsetY - guideDeltaY / 2)\n\n      const offsetXMax = -Math.abs(Math.round(container.width - width))\n      const offsetYMax = -Math.abs(Math.round(container.height - height))\n      const guideOffsetXMax = Math.round(minimap.width - guideWidth)\n      const guideOffsetYMax = Math.round(minimap.height - guideHeight)\n\n      if (image.initialWidth > width && image.initialHeight > height) {\n        this.setState((prevState) => ({\n          image: {\n            ...prevState.image,\n            width,\n            height,\n            scale,\n            offsetX: offsetX >= 0 || container.width > width\n              ? 0\n              : image.offsetX <= offsetXMax\n                ? offsetXMax\n                : offsetX,\n            offsetY: (container.height > height)\n              ? container.height / 2 - height / 2\n              : offsetY >= 0\n                ? 0\n                : image.offsetY < offsetYMax\n                  ? offsetYMax\n                  : offsetY\n          },\n          minimap: {\n            ...prevState.minimap,\n            guideWidth,\n            guideHeight,\n            offsetX: guideOffsetX <= 0\n              ? 0\n              : minimap.offsetX < guideOffsetXMax\n                ? guideOffsetX\n                : guideOffsetXMax,\n            offsetY: guideOffsetY <= 0 || height < container.height\n              ? 0\n              : minimap.offsetY < guideOffsetYMax\n                ? guideOffsetY\n                : guideOffsetYMax\n          },\n          draggable: scale > 1\n        }))\n      }\n\n      // Reset image position\n      if (scale === 1) {\n        this.setState((prevState) => ({\n          image: {\n            ...prevState.image,\n            offsetX: 0,\n            offsetY: container.height / 2 - height / 2\n          },\n          minimap: {\n            ...prevState.minimap,\n            offsetX: 0,\n            offsetY: 0\n          }\n        }))\n      }\n    }\n  }\n\n  requestFullscreen = (element) => {\n    if (element.requestFullscreen) {\n      element.requestFullscreen()\n    } else if (element.mozRequestFullScreen) {\n      element.mozRequestFullScreen()\n    } else if (element.webkitRequestFullscreen) {\n      element.webkitRequestFullscreen()\n    } else if (element.msRequestFullscreen) {\n      element.msRequestFullscreen()\n    }\n  }\n\n  exitFullscreen = () => {\n    if (document.exitFullscreen) {\n      document.exitFullscreen()\n    } else if (document.mozCancelFullScreen) {\n      document.mozCancelFullScreen()\n    } else if (document.webkitExitFullscreen) {\n      document.webkitExitFullscreen()\n    } else if (document.msExitFullscreen) {\n      document.msExitFullscreen()\n    }\n  }\n\n  render = () => {\n    const { src, alt, hotspots, background } = this.props\n    const {\n      container,\n      image,\n      minimap,\n      fullscreen,\n      dragging,\n      hideFullscreenControl,\n      hideZoomControls,\n      hideMinimap,\n      draggable\n    } = this.state\n    const imageLoaded = image.initialWidth && image.initialHeight\n\n    const containerStyle = {\n      width: '100%',\n      height: '100%',\n      position: 'relative',\n      overflow: 'hidden',\n      textAlign: 'center',\n      background: background || container.background\n    }\n\n    const imageStyle = {\n      position: 'relative',\n      left: image.offsetX,\n      top: image.offsetY\n    }\n\n    const hotspotsStyle = {\n      position: 'absolute',\n      top: image.offsetY,\n      left: image.offsetX,\n      right: (image.offsetX >= 0) ? 0 : 'auto',\n      margin: 'auto',\n      pointerEvents: 'none'\n    }\n\n    const topControlsStyle = {\n      position: 'absolute',\n      top: 10,\n      right: 10,\n      pointerEvents: this.state.dragging ? 'none' : 'auto'\n    }\n\n    const bottomControlsStyle = {\n      position: 'absolute',\n      bottom: 10,\n      right: 10,\n      pointerEvents: this.state.dragging ? 'none' : 'auto'\n    }\n\n    const buttonStyle = {\n      width: '25px',\n      height: '25px',\n      border: 'none',\n      background: '#fff',\n      boxShadow: '0px 0px 2px 0px rgba(0,0,0,0.5)'\n    }\n\n    const minimapStyle = {\n      width: minimap.width,\n      height: minimap.height,\n      position: 'absolute',\n      display: 'block',\n      bottom: 10,\n      left: 10,\n      background: '#fff',\n      boxShadow: '0px 0px 2px 0px rgba(0,0,0,0.5)',\n      pointerEvents: 'none'\n    }\n\n    const guideStyle = {\n      width: minimap.guideWidth,\n      height: minimap.guideHeight,\n      position: 'absolute',\n      display: 'block',\n      left: minimap.offsetX,\n      top: minimap.offsetY,\n      border: '1px solid rgba(64, 139, 252, 0.8)',\n      background: 'rgba(64, 139, 252, 0.1)',\n      pointerEvents: 'none'\n    }\n\n    if (imageLoaded) {\n      if (container.orientation === 'landscape') {\n        imageStyle.height = image.height\n      } else {\n        imageStyle.width = image.width\n      }\n\n      if (image.orientation === 'landscape') {\n        hotspotsStyle.width = image.width\n        hotspotsStyle.height = image.width / image.ratio\n      } else {\n        hotspotsStyle.width = image.height / image.ratio\n        hotspotsStyle.height = image.height\n      }\n    }\n\n    return (\n      <div\n        ref={this.container}\n        style={containerStyle}\n        onMouseOut={event => {\n          if (dragging) {\n            this.stopDrag(event)\n          }\n        }}\n        onBlur={event => {\n          if (dragging) {\n            this.stopDrag(event)\n          }\n        }}\n      >\n        {\n          src &&\n            <img\n              src={src}\n              alt={alt}\n              onLoad={this.handleOnImageLoad}\n              style={imageStyle}\n              onMouseDown={event => {\n                if (!hideZoomControls && draggable) {\n                  this.startDrag(event, 'image')\n                }\n              }}\n              onMouseMove={event => {\n                if (!hideZoomControls && dragging) {\n                  this.whileDrag(event)\n                }\n              }}\n              onMouseUp={event => {\n                if (dragging) {\n                  this.stopDrag(event)\n                }\n              }}\n            />\n        }\n        {\n          hotspots &&\n            <div style={hotspotsStyle}>\n              {\n              hotspots.map((hotspot, i) => <Hotspot key={i} {...hotspot} />)\n            }\n            </div>\n        }\n        {\n          !hideFullscreenControl &&\n            <div style={topControlsStyle}>\n              <button style={buttonStyle} onClick={() => this.toggleFullscreen()}>\n                {fullscreen ? 'X' : 'FS'}\n              </button>\n            </div>\n        }\n        {\n          !hideZoomControls &&\n            <>\n              <div style={bottomControlsStyle}>\n                {\n                  draggable &&\n                    <>\n                      <button style={buttonStyle} onClick={() => this.zoom(1)}>Fit</button>\n                      <br />\n                      <br />\n                    </>\n                }\n                <button style={buttonStyle} onClick={() => this.zoom(image.scale + 1)}>+</button>\n                <br />\n                <button style={buttonStyle} onClick={() => this.zoom(image.scale - 1)}>-</button>\n              </div>\n              {\n                !hideMinimap &&\n                  <div style={minimapStyle}>\n                    {src &&\n                      <img src={src} width={minimapStyle.width} height={minimapStyle.height} />}\n                    <div style={guideStyle} />\n                  </div>\n              }\n            </>\n        }\n      </div>\n    )\n  }\n}\n\nImageHotspots.propTypes = {\n  src: PropTypes.string,\n  alt: PropTypes.string,\n  hotspots: PropTypes.array\n}\n\nexport default ImageHotspots\n"],"names":["ImageHotspots","React","Component","props","componentDidMount","hideFullscreenControl","hideZoomControls","hideMinimap","hotspots","background","offsetWidth","width","offsetHeight","height","container","current","orientation","ratio","setState","window","addEventListener","onWindowResize","componentWillUnmount","removeEventListener","startDrag","event","element","cursorX","clientX","cursorY","clientY","state","dragging","preventDefault","whileDrag","image","minimap","deltaX","deltaY","newOffsetX","offsetX","newOffsetY","offsetY","stopDrag","offsetXMax","Math","abs","offsetYMax","handleOnImageLoad","target","initialWidth","initialHeight","resizable","prevState","scale","initialSize","guideWidth","guideHeight","draggable","zoom","toggleFullscreen","fullscreen","requestFullscreen","exitFullscreen","round","guideDeltaX","guideDeltaY","guideOffsetX","guideOffsetY","guideOffsetXMax","guideOffsetYMax","mozRequestFullScreen","webkitRequestFullscreen","msRequestFullscreen","document","mozCancelFullScreen","webkitExitFullscreen","msExitFullscreen","render","src","alt","imageLoaded","containerStyle","position","overflow","textAlign","imageStyle","left","top","hotspotsStyle","right","margin","pointerEvents","topControlsStyle","bottomC