UNPKG

react-image-hotspots

Version:
466 lines (465 loc) 67.2 kB
"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