UNPKG

antd-react-cropper

Version:

An image cropper for Ant Design Upload using React Cropper.

421 lines (363 loc) 18.6 kB
'use strict'; Object.defineProperty(exports, '__esModule', { value: true }); var React = require('react'); var antd = require('antd'); var Cropper = require('react-cropper'); function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; } var React__default = /*#__PURE__*/_interopDefaultLegacy(React); var Cropper__default = /*#__PURE__*/_interopDefaultLegacy(Cropper); function styleInject(css, ref) { if ( ref === void 0 ) ref = {}; var insertAt = ref.insertAt; if (!css || typeof document === 'undefined') { return; } var head = document.head || document.getElementsByTagName('head')[0]; var style = document.createElement('style'); style.type = 'text/css'; if (insertAt === 'top') { if (head.firstChild) { head.insertBefore(style, head.firstChild); } else { head.appendChild(style); } } else { head.appendChild(style); } if (style.styleSheet) { style.styleSheet.cssText = css; } else { style.appendChild(document.createTextNode(css)); } } var css_248z = "/*!\n * Cropper.js v1.5.12\n * https://fengyuanchen.github.io/cropperjs\n *\n * Copyright 2015-present Chen Fengyuan\n * Released under the MIT license\n *\n * Date: 2021-06-12T08:00:11.623Z\n */\n\n.cropper-container {\n direction: ltr;\n font-size: 0;\n line-height: 0;\n position: relative;\n -ms-touch-action: none;\n touch-action: none;\n -webkit-user-select: none;\n -moz-user-select: none;\n -ms-user-select: none;\n user-select: none;\n}\n\n.cropper-container img {\n display: block;\n height: 100%;\n image-orientation: 0deg;\n max-height: none !important;\n max-width: none !important;\n min-height: 0 !important;\n min-width: 0 !important;\n width: 100%;\n}\n\n.cropper-wrap-box,\n.cropper-canvas,\n.cropper-drag-box,\n.cropper-crop-box,\n.cropper-modal {\n bottom: 0;\n left: 0;\n position: absolute;\n right: 0;\n top: 0;\n}\n\n.cropper-wrap-box,\n.cropper-canvas {\n overflow: hidden;\n}\n\n.cropper-drag-box {\n background-color: #fff;\n opacity: 0;\n}\n\n.cropper-modal {\n background-color: #000;\n opacity: 0.5;\n}\n\n.cropper-view-box {\n display: block;\n height: 100%;\n outline: 1px solid #39f;\n outline-color: rgba(51, 153, 255, 0.75);\n overflow: hidden;\n width: 100%;\n}\n\n.cropper-dashed {\n border: 0 dashed #eee;\n display: block;\n opacity: 0.5;\n position: absolute;\n}\n\n.cropper-dashed.dashed-h {\n border-bottom-width: 1px;\n border-top-width: 1px;\n height: calc(100% / 3);\n left: 0;\n top: calc(100% / 3);\n width: 100%;\n}\n\n.cropper-dashed.dashed-v {\n border-left-width: 1px;\n border-right-width: 1px;\n height: 100%;\n left: calc(100% / 3);\n top: 0;\n width: calc(100% / 3);\n}\n\n.cropper-center {\n display: block;\n height: 0;\n left: 50%;\n opacity: 0.75;\n position: absolute;\n top: 50%;\n width: 0;\n}\n\n.cropper-center::before,\n.cropper-center::after {\n background-color: #eee;\n content: ' ';\n display: block;\n position: absolute;\n}\n\n.cropper-center::before {\n height: 1px;\n left: -3px;\n top: 0;\n width: 7px;\n}\n\n.cropper-center::after {\n height: 7px;\n left: 0;\n top: -3px;\n width: 1px;\n}\n\n.cropper-face,\n.cropper-line,\n.cropper-point {\n display: block;\n height: 100%;\n opacity: 0.1;\n position: absolute;\n width: 100%;\n}\n\n.cropper-face {\n background-color: #fff;\n left: 0;\n top: 0;\n}\n\n.cropper-line {\n background-color: #39f;\n}\n\n.cropper-line.line-e {\n cursor: ew-resize;\n right: -3px;\n top: 0;\n width: 5px;\n}\n\n.cropper-line.line-n {\n cursor: ns-resize;\n height: 5px;\n left: 0;\n top: -3px;\n}\n\n.cropper-line.line-w {\n cursor: ew-resize;\n left: -3px;\n top: 0;\n width: 5px;\n}\n\n.cropper-line.line-s {\n bottom: -3px;\n cursor: ns-resize;\n height: 5px;\n left: 0;\n}\n\n.cropper-point {\n background-color: #39f;\n height: 5px;\n opacity: 0.75;\n width: 5px;\n}\n\n.cropper-point.point-e {\n cursor: ew-resize;\n margin-top: -3px;\n right: -3px;\n top: 50%;\n}\n\n.cropper-point.point-n {\n cursor: ns-resize;\n left: 50%;\n margin-left: -3px;\n top: -3px;\n}\n\n.cropper-point.point-w {\n cursor: ew-resize;\n left: -3px;\n margin-top: -3px;\n top: 50%;\n}\n\n.cropper-point.point-s {\n bottom: -3px;\n cursor: s-resize;\n left: 50%;\n margin-left: -3px;\n}\n\n.cropper-point.point-ne {\n cursor: nesw-resize;\n right: -3px;\n top: -3px;\n}\n\n.cropper-point.point-nw {\n cursor: nwse-resize;\n left: -3px;\n top: -3px;\n}\n\n.cropper-point.point-sw {\n bottom: -3px;\n cursor: nesw-resize;\n left: -3px;\n}\n\n.cropper-point.point-se {\n bottom: -3px;\n cursor: nwse-resize;\n height: 20px;\n opacity: 1;\n right: -3px;\n width: 20px;\n}\n\n@media (min-width: 768px) {\n .cropper-point.point-se {\n height: 15px;\n width: 15px;\n }\n}\n\n@media (min-width: 992px) {\n .cropper-point.point-se {\n height: 10px;\n width: 10px;\n }\n}\n\n@media (min-width: 1200px) {\n .cropper-point.point-se {\n height: 5px;\n opacity: 0.75;\n width: 5px;\n }\n}\n\n.cropper-point.point-se::before {\n background-color: #39f;\n bottom: -50%;\n content: ' ';\n display: block;\n height: 200%;\n opacity: 0;\n position: absolute;\n right: -50%;\n width: 200%;\n}\n\n.cropper-invisible {\n opacity: 0;\n}\n\n.cropper-bg {\n background-image: url('');\n}\n\n.cropper-hide {\n display: block;\n height: 0;\n position: absolute;\n width: 0;\n}\n\n.cropper-hidden {\n display: none !important;\n}\n\n.cropper-move {\n cursor: move;\n}\n\n.cropper-crop {\n cursor: crosshair;\n}\n\n.cropper-disabled .cropper-drag-box,\n.cropper-disabled .cropper-face,\n.cropper-disabled .cropper-line,\n.cropper-disabled .cropper-point {\n cursor: not-allowed;\n}\n"; styleInject(css_248z); function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; } function _objectSpread2(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { _defineProperty(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; } function _defineProperty(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 _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); } function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; } function _iterableToArrayLimit(arr, i) { var _i = arr == null ? null : typeof Symbol !== "undefined" && arr[Symbol.iterator] || arr["@@iterator"]; if (_i == null) return; var _arr = []; var _n = true; var _d = false; var _s, _e; try { for (_i = _i.call(arr); !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"] != null) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); } function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) arr2[i] = arr[i]; return arr2; } function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } var css_248z$1 = ".container {\n display: flex;\n width: 100%;\n height: 100%;\n}\n.img-cropper {\n width: 100%;\n height: 100%;\n border-radius: 6px;\n}\n.img-cropper .cropper-container,\n.img-cropper .cropper-crop-box,\n.img-cropper .cropper-drag-box,\n.img-cropper .cropper-wrap-box {\n border-radius: 6px !important;\n}\n.slider-container {\n display: flex;\n align-items: center;\n justify-self: center;\n width: 100%;\n margin: 4px 0px;\n -webkit-user-select: none;\n -moz-user-select: none;\n -ms-user-select: none;\n user-select: none;\n}\n.slider-container .slider {\n flex: 1;\n margin-right: 20px !important;\n margin-left: 20px !important;\n}\n.preview-container {\n display: flex;\n align-items: center;\n justify-content: center;\n width: 100%;\n height: '100%';\n overflow: hidden;\n border-radius: 6px;\n position: 'relative';\n}\n.preview-container .preview-content {\n float: left;\n width: '100%';\n overflow: hidden;\n}\n.divider {\n margin-top: 4px !important;\n margin-bottom: 16px !important;\n}\n.flex {\n display: flex;\n}\n.flex-center {\n display: flex !important;\n align-items: center !important;\n justify-content: center !important;\n}\n.modal-footer {\n display: flex;\n}\n.action-btn {\n display: flex !important;\n align-items: center !important;\n justify-content: center !important;\n width: 20px;\n padding: 0 !important;\n}\n"; styleInject(css_248z$1); var maxZoom = 10; var ImageCropper = function ImageCropper(props) { var aspect = props.aspect, file = props.file, onCropped = props.onCropped, previewMaxHeight = props.previewMaxHeight, title = props.title; var screens = antd.Grid.useBreakpoint(); var _useState = React.useState(), _useState2 = _slicedToArray(_useState, 2), src = _useState2[0], setSrc = _useState2[1]; var _useState3 = React.useState(), _useState4 = _slicedToArray(_useState3, 2), cropper = _useState4[0], setCropper = _useState4[1]; var _useState5 = React.useState(0), _useState6 = _slicedToArray(_useState5, 2), zoom = _useState6[0], setZoom = _useState6[1]; var _useState7 = React.useState(0), _useState8 = _slicedToArray(_useState7, 2), rotate = _useState8[0], setRotate = _useState8[1]; var _useState9 = React.useState(), _useState10 = _slicedToArray(_useState9, 2), canvasData = _useState10[0], setCanvasData = _useState10[1]; React.useEffect(function () { if (file && file.originFileObj) getFileUrl(file.originFileObj, setSrc); }, [file]); React.useEffect(function () { if (!src) { setZoom(0); setRotate(0); setCanvasData(undefined); setCropper(undefined); } }, [src]); React.useEffect(function () { if (cropper) cropper.rotateTo(rotate); }, [rotate]); var onZoomIn = function onZoomIn() { var to = zoom + 1; if (to > maxZoom) to = maxZoom; setZoom(to); onZoom(to); }; var onZoomOut = function onZoomOut() { var to = zoom - 1; if (to <= 0) to = 0; setZoom(to); onZoom(to); }; var onZoomSlide = function onZoomSlide(value) { setZoom(value); onZoom(value); }; var onRotateLeft = function onRotateLeft() { var to = rotate - 15; if (to <= -180) to = -180; setRotate(to); }; var onRotateRight = function onRotateRight() { var to = rotate + 15; if (to >= 180) to = 180; setRotate(to); }; var onCrop = function onCrop() { if (typeof cropper !== 'undefined') { var loader; if (props.showLoader) loader = antd.message.loading('Processing image...', 0); var cropped = cropper.getCroppedCanvas(); var thumbUrl = cropped.toDataURL('image/jpeg'); cropped.toBlob(function (blob) { if (!blob) return; var newFile = new File([blob], file.name); var data = _objectSpread2(_objectSpread2({}, file), {}, { thumbUrl: thumbUrl, originFileObj: newFile }); if (loader) loader(); onCropped === null || onCropped === void 0 ? void 0 : onCropped(data); }, props.exportFileType); } setSrc(undefined); }; var onCancel = function onCancel() { setSrc(undefined); if (props.onCancel) props.onCancel(); }; var onZoom = function onZoom(value) { if (!cropper) return; if (!canvasData) setCanvasData(cropper.getCanvasData()); if (value > maxZoom) return; if (value <= 0) { cropper.setCanvasData(canvasData); } else if (value > zoom) { var diff = value - zoom; cropper.zoom(diff * 0.2); } else { var _diff = zoom - value; cropper.zoom(_diff * -0.2); } }; return /*#__PURE__*/React__default['default'].createElement(antd.Modal, { title: title, zIndex: props.zIndex, centered: props.centered, width: props.width, closable: props.closable || false, bodyStyle: props.bodyStyle || { paddingBottom: 0 }, maskClosable: props.maskClosable, destroyOnClose: true, visible: !!src, onCancel: onCancel, footer: props.customFooter ? props.customFooter({ onCrop: onCrop, onCancel: onCancel }) : /*#__PURE__*/React__default['default'].createElement("div", { className: "modal-footer" }, /*#__PURE__*/React__default['default'].createElement(antd.Button, { onClick: onCancel }, props.cancelText), /*#__PURE__*/React__default['default'].createElement(antd.Button, { block: true, type: "primary", onClick: onCrop }, props.cropText)) }, /*#__PURE__*/React__default['default'].createElement(antd.Row, { gutter: 12 }, /*#__PURE__*/React__default['default'].createElement(antd.Col, { xs: 24, sm: 16 }, /*#__PURE__*/React__default['default'].createElement("div", null, src && /*#__PURE__*/React__default['default'].createElement(Cropper__default['default'], { className: "img-cropper ".concat(props.cropperClassName || ''), style: props.cropperStyle, dragMode: "move", aspectRatio: aspect, preview: ".img-preview", src: src, viewMode: 0, background: true, center: true, autoCropArea: 1, scalable: false, movable: true, onInitialized: setCropper, checkOrientation: false }), /*#__PURE__*/React__default['default'].createElement("div", { className: "slider-container" }, props.zoomOutButton ? props.zoomOutButton(onZoomOut) : /*#__PURE__*/React__default['default'].createElement(antd.Button, { className: "action-btn", type: "link", onClick: onZoomOut }, /*#__PURE__*/React__default['default'].createElement("span", { style: { fontSize: 20 } }, "-")), /*#__PURE__*/React__default['default'].createElement(antd.Slider, { className: "slider", value: zoom, min: 0, max: maxZoom, onChange: onZoomSlide }), props.zoomInButton ? props.zoomInButton(onZoomIn) : /*#__PURE__*/React__default['default'].createElement(antd.Button, { className: "action-btn", type: "link", onClick: onZoomIn }, /*#__PURE__*/React__default['default'].createElement("span", { style: { fontSize: 20 } }, "+"))), /*#__PURE__*/React__default['default'].createElement("div", { className: "slider-container" }, props.rotateLeftButton ? props.rotateLeftButton(onRotateLeft) : /*#__PURE__*/React__default['default'].createElement(antd.Button, { className: "action-btn", type: "link", onClick: onRotateLeft }, /*#__PURE__*/React__default['default'].createElement("span", { style: { fontSize: 15 } }, "\u21BA")), /*#__PURE__*/React__default['default'].createElement("div", { style: { flex: 1 }, onDoubleClick: function onDoubleClick() { return setRotate(0); } }, /*#__PURE__*/React__default['default'].createElement(antd.Slider, { className: "slider", value: rotate, min: -180, max: 180, onChange: setRotate })), props.rotateRightButton ? props.rotateRightButton(onRotateRight) : /*#__PURE__*/React__default['default'].createElement(antd.Button, { className: "action-btn", type: "link", onClick: onRotateRight }, /*#__PURE__*/React__default['default'].createElement("span", { style: { fontSize: 15 } }, "\u21BB"))))), /*#__PURE__*/React__default['default'].createElement(antd.Col, { xs: 24, sm: 8 }, screens.xs && /*#__PURE__*/React__default['default'].createElement(antd.Divider, { className: "divider" }), /*#__PURE__*/React__default['default'].createElement("div", { className: "preview-container ".concat(props.previewClassName || ''), style: _objectSpread2(_objectSpread2({}, props.previewStyle), {}, { maxHeight: previewMaxHeight, backgroundColor: props.previewBackground }) }, /*#__PURE__*/React__default['default'].createElement("div", { className: "img-preview", style: { width: '100%', height: previewMaxHeight, float: 'left', overflow: 'hidden', background: '#000' } }))))); }; var getFileUrl = function getFileUrl(file, callback) { return new Promise(function (resolve) { var reader = new FileReader(); reader.onload = function () { resolve(reader.result); callback === null || callback === void 0 ? void 0 : callback(reader.result); }; reader.readAsDataURL(file); }); }; ImageCropper.defaultProps = { centered: true, width: 500, previewMaxHeight: 150, previewBackground: 'rgba(212, 212, 216)', cropText: 'Crop', cancelText: 'Cancel', showLoader: true, exportFileType: 'image/jpeg' }; exports.ImageCropper = ImageCropper; exports.default = ImageCropper;