react-sprucebot
Version:
React components for your Sprucebot Skill đĒđŧ
569 lines (532 loc) âĸ 14.3 kB
JavaScript
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;
}
}
`