UNPKG

react-sprucebot

Version:

React components for your Sprucebot Skill đŸ’ĒđŸŧ

569 lines (532 loc) â€ĸ 14.3 kB
import React, { Component } from 'react' import ReactCrop, { getPixelCrop } from 'react-image-crop' import BotText from '../BotText/BotText' import Button from '../Button/Button' import ExecutionEnvironment from 'exenv' import Loader from '../Loader/Loader' import PropTypes from 'prop-types' import SubmitWrapper from '../SubmitWrapper/SubmitWrapper' import getOrientedImage from 'exif-orientation-image' import styled from 'styled-components' if (ExecutionEnvironment.canUseDOM) { require('blueimp-canvas-to-blob') } export default class ImageCropper extends Component { constructor(props) { super(props) this.state = { errorMessage: '', base64Image: props.base64Image, crop: props.crop, pixelCrop: props.prop, changed: false, loading: !!props.src, // if there is an image src being passed, we have to actually fetch it tapToCrop: props.tapToCrop, uploading: false, newFile: false, type: props.src ? `image/${props.src.split('.').pop()}` : false, aspect: props.crop.aspect } } initiateFileUpload() { if (this.state.uploading) { return } this.input.click() } componentDidMount() { // is browser out-to-date, Mayura??? if (typeof FileReader === 'undefined') { this.setState({ errorMessage: this.props.outOfDateBrowserMessage }) } else { // setup file reader this.reader = new FileReader() this.reader.onload = this.onFileReaderLoadImage.bind(this) this.reader.onerror = this.onFileReaderLoadImageFail.bind(this) } } onChange(e) { const file = e.target.files[0] if (!file.type.match('image.*')) { alert(this.props.badImageMessage) return } getOrientedImage(file, (err, canvas) => { if (!err) { this.setState({ changed: true, newFile: true }) canvas.toBlob(blob => this.reader.readAsDataURL(blob)) } }) } onFileReaderLoadImage(e) { const base64 = e.target.result const type = base64.substr(5, base64.search(';') - 5) this.setState({ loading: false, tapToCrop: false, errorMessage: false, base64Image: base64, type: type }) } onFileReaderLoadImageFail(err) { console.error(err) this.setState({ errorMessage: this.props.uploadImageFailedMessage }) } onCropChange(crop, pixelCrop) { const maxWidth = window.innerWidth - 20 const maxHeight = window.innerHeight - 20 const x = window.event.x const y = window.event.y if (x > maxWidth || x < 20 || y < 20 || y > maxHeight) { this.cropper.onDocMouseTouchEnd() } this.setState({ crop, pixelCrop, changed: true }) } onImageLoadedFromCropper(image) { if (!this.cropper) { // this can happen when the cropper is hidden, then shown return } else { const crop = this.state.crop const pixelCrop = getPixelCrop(image, crop) const widthHeight = image.height < image.width ? image.height / 2 : image.width / 2 const width = (widthHeight / image.width) * 100 const height = (widthHeight / image.height) * 100 crop.width = width crop.height = height crop.x = width >= height ? width / 2 : width crop.y = width <= height ? height / 2 : height if (this.state.aspect) { crop.aspect = this.state.aspect } this.setState({ crop, pixelCrop, loading: false }) } } componentWillReceiveProps(nextProps) { if (nextProps.src !== this.props.src) { this.setState({ type: `image/${nextProps.src.split('.').pop()}`, base64Image: false, newFile: true, src: nextProps.src }) } } hideBlock() { this.setState({ tapToCrop: false, crop: this.state.crop, changed: true }) } async onSave() { const { pixelCrop, type } = this.state if (this.state.uploading) { return } if (!type) { this.setState({ errorMessage: this.props.badImageMessage }) return } try { this.setState({ uploading: true }) const image = await new Promise((resolve, reject) => { const image = new Image() image.onload = () => { resolve(image) } image.onerror = function(err) { reject(err) } image.src = this.cropper.imageRef.src }) const canvas = document.createElement('canvas') canvas.width = pixelCrop.width canvas.height = pixelCrop.height const ctx = canvas.getContext('2d') ctx.drawImage( image, pixelCrop.x, pixelCrop.y, pixelCrop.width, pixelCrop.height, 0, 0, pixelCrop.width, pixelCrop.height ) const cropped = canvas.toDataURL(type) await this.props.onSave(cropped, type) // reset things how they were this.setState({ tapToCrop: this.props.tapToCrop, changed: false, newFile: false, base64Image: cropped }) } catch (err) { console.error(err) this.setState({ errorMessage: this.props.uploadImageFailedMessage }) } this.setState({ uploading: false }) } cancel() { this.setState({ tapToCrop: this.props.tapToCrop, crop: this.props.crop, changed: false }) } render() { const { accept, uploadButtonText, uploadNewButtonText, saveButtonText, tapToCropButtonText, cancelButtonText, src } = this.props const { uploading, base64Image, crop, loading, changed, tapToCrop, errorMessage, newFile } = this.state const cropSrc = base64Image || src return ( <div className="image_cropper"> {errorMessage && <BotText>{errorMessage}</BotText>} {loading && <Loader />} {!errorMessage && cropSrc && ( <StyledReactCrop loading={loading} tapToCrop={tapToCrop}> <ReactCrop ref={cropper => (this.cropper = cropper)} keepSelection={true} onImageLoaded={this.onImageLoadedFromCropper.bind(this)} src={cropSrc} crop={crop} onChange={this.onCropChange.bind(this)} /> {tapToCrop && ( <div className="block"> {!loading && ( <Button onClick={this.hideBlock.bind(this)}> {tapToCropButtonText} </Button> )} </div> )} </StyledReactCrop> )} <input style={{ display: 'none' }} type="file" ref={input => { this.input = input }} accept={accept} onChange={this.onChange.bind(this)} /> {!loading && ( <SubmitWrapper> {changed && !errorMessage && ( <Button busy={uploading} onClick={this.onSave.bind(this)} primary > {saveButtonText} </Button> )} {changed && !newFile && !errorMessage && ( <Button busy={uploading} onClick={this.cancel.bind(this)} secondary > {cancelButtonText} </Button> )} <Button busy={uploading} alt onClick={this.initiateFileUpload.bind(this)} > {src ? uploadNewButtonText : uploadButtonText} </Button> </SubmitWrapper> )} </div> ) } } ImageCropper.propTypes = { base64Image: PropTypes.string, imageUrl: PropTypes.string, onSave: PropTypes.func.isRequired, badImageMessage: PropTypes.any.isRequired, outOfDateBrowserMessage: PropTypes.any.isRequired, uploadImageFailedMessage: PropTypes.any.isRequired, loadingImageFailedMessage: PropTypes.any.isRequired, uploadButtonText: PropTypes.any.isRequired, uploadNewButtonText: PropTypes.any.isRequired, tapToCropButtonText: PropTypes.any.isRequired, saveButtonText: PropTypes.any.isRequired, cancelButtonText: PropTypes.any.isRequired, accept: PropTypes.string.isRequired, crop: PropTypes.object.isRequired, tapToCrop: PropTypes.bool } ImageCropper.defaultProps = { accept: 'image/*', loadingImageFailedMessage: "Uh man, I couldn't load your image.", badImageMessage: 'Bad upload! You gotta select an image.', outOfDateBrowserMessage: 'You gotta update your browser to upload and crop images. â˜šī¸', uploadImageFailedMessage: "So, this is embarrassing, but I could not upload that file and couldn't tell you why. 😞", uploadButtonText: 'Upload Image', uploadNewButtonText: 'Upload Different Image', saveButtonText: 'Save Changes', tapToCropButtonText: 'Tap to Re-Crop', cancelButtonText: 'Cancel Crop', tapToCrop: false, crop: {} } const StyledReactCrop = styled.div` position: relative; opacity: ${props => (props.loading ? 0 : 1)}; .block { position: absolute; left: 0px; top: 0px; right: 0px; bottom: 0px; background-color: rgba(0, 0, 0, 0.6); line-height: 100%; text-align: center; display: flex; justify-content: center; button { height: 50px; width: 150px; align-self: center; } } .ReactCrop { position: relative; display: inline-block; cursor: crosshair; overflow: hidden; width: 100%; } .ReactCrop:focus { outline: none; } .ReactCrop--disabled { cursor: inherit; } .ReactCrop__image { display: block; width: 100%; } .ReactCrop--crop-invisible .ReactCrop__image { opacity: 0.5; } .ReactCrop__crop-selection { position: absolute; top: 0; left: 0; transform: translate3d(0, 0, 0); box-sizing: border-box; cursor: move; box-shadow: 0 0 0 9999em rgba(0, 0, 0, 0.5); border: 1px solid; border-image-source: url(''); border-image-slice: 1; border-image-repeat: repeat; opacity: ${props => (props.tapToCrop ? 0 : 1)}; } .ReactCrop--disabled .ReactCrop__crop-selection { cursor: inherit; } .ReactCrop__drag-handle { position: absolute; width: 9px; height: 9px; background-color: rgba(0, 0, 0, 0.2); border: 1px solid rgba(255, 255, 255, 0.7); box-sizing: border-box; outline: 1px solid transparent; } .ReactCrop .ord-nw { top: 0; left: 0; margin-top: -5px; margin-left: -5px; cursor: nw-resize; } .ReactCrop .ord-n { top: 0; left: 50%; margin-top: -5px; margin-left: -5px; cursor: n-resize; } .ReactCrop .ord-ne { top: 0; right: 0; margin-top: -5px; margin-right: -5px; cursor: ne-resize; } .ReactCrop .ord-e { top: 50%; right: 0; margin-top: -5px; margin-right: -5px; cursor: e-resize; } .ReactCrop .ord-se { bottom: 0; right: 0; margin-bottom: -5px; margin-right: -5px; cursor: se-resize; } .ReactCrop .ord-s { bottom: 0; left: 50%; margin-bottom: -5px; margin-left: -5px; cursor: s-resize; } .ReactCrop .ord-sw { bottom: 0; left: 0; margin-bottom: -5px; margin-left: -5px; cursor: sw-resize; } .ReactCrop .ord-w { top: 50%; left: 0; margin-top: -5px; margin-left: -5px; cursor: w-resize; } .ReactCrop__disabled .ReactCrop__drag-handle { cursor: inherit; } .ReactCrop__drag-bar { position: absolute; } .ReactCrop__drag-bar.ord-n { top: 0; left: 0; width: 100%; height: 6px; margin-top: -4px; } .ReactCrop__drag-bar.ord-e { right: 0; top: 0; width: 6px; height: 100%; margin-right: -4px; } .ReactCrop__drag-bar.ord-s { bottom: 0; left: 0; width: 100%; height: 6px; margin-bottom: -4px; } .ReactCrop__drag-bar.ord-w { top: 0; left: 0; width: 6px; height: 100%; margin-left: -4px; } .ReactCrop--new-crop .ReactCrop__drag-bar, .ReactCrop--new-crop .ReactCrop__drag-handle, .ReactCrop--fixed-aspect .ReactCrop__drag-bar { display: none; } .ReactCrop--fixed-aspect .ReactCrop__drag-handle.ord-n, .ReactCrop--fixed-aspect .ReactCrop__drag-handle.ord-e, .ReactCrop--fixed-aspect .ReactCrop__drag-handle.ord-s, .ReactCrop--fixed-aspect .ReactCrop__drag-handle.ord-w { display: none; } @media (max-width: 768px) { .ReactCrop__drag-handle { width: 17px; height: 17px; } .ReactCrop .ord-nw { margin-top: -9px; margin-left: -9px; } .ReactCrop .ord-n { margin-top: -9px; margin-left: -9px; } .ReactCrop .ord-ne { margin-top: -9px; margin-right: -9px; } .ReactCrop .ord-e { margin-top: -9px; margin-right: -9px; } .ReactCrop .ord-se { margin-bottom: -9px; margin-right: -9px; } .ReactCrop .ord-s { margin-bottom: -9px; margin-left: -9px; } .ReactCrop .ord-sw { margin-bottom: -9px; margin-left: -9px; } .ReactCrop .ord-w { margin-top: -9px; margin-left: -9px; } .ReactCrop__drag-bar.ord-n { height: 14px; margin-top: -12px; } .ReactCrop__drag-bar.ord-e { width: 14px; margin-right: -12px; } .ReactCrop__drag-bar.ord-s { height: 14px; margin-bottom: -12px; } .ReactCrop__drag-bar.ord-w { width: 14px; margin-left: -12px; } } `