Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Transform prop and flip demo #150

Merged
merged 10 commits into from
Jun 24, 2020
Merged
1 change: 1 addition & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -108,6 +108,7 @@ This component requires some styles to be available in the document. By default,
| `onZoomChange` | zoom => void | | Called everytime the zoom is changed. Use it to update your `zoom` state. |
| `onRotationChange` | rotation => void | | Called everytime the rotation is changed (with mobile gestures). Use it to update your `rotation` state. |
| [`onCropComplete`](#onCropCompleteProp) | Function | | Called when the user stops moving the media or stops zooming. It will be passed the corresponding cropped area on the media in percentages and pixels |
| `transform` | string | | CSS transform to apply to the image in the editor. Defaults to `translate(${crop.x}px, ${crop.y}px) rotate(${rotation}deg) scale(${zoom})` with variables being pulled from props. |
| `style` | `{ containerStyle: object, mediaStyle: object, cropAreaStyle: object }` | | Custom styles to be used with the Cropper. Styles passed via the style prop are merged with the defaults. |
| `classes` | `{ containerClassName: string, mediaClassName: string, cropAreaClassName: string }` | | Custom class names to be used with the Cropper. Classes passed via the classes prop are merged with the defaults. If you have CSS specificity issues, you should probably use the `disableAutomaticStylesInjection` prop. |
| `mediaProps` | object | | The properties you want to apply to the media tag (<img /> or <video /> depending on your media) |
Expand Down
6 changes: 4 additions & 2 deletions docs/src/components/Demo/cropImage.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,8 @@ export const createImage = (url: string): Promise<HTMLImageElement> =>
export default async function getCroppedImg(
imageSrc: string,
pixelCrop: Area,
rotation = 0
rotation = 0,
flip = { horizontal: false, vertical: false }
): Promise<string | null> {
const image = await createImage(imageSrc)
const canvas = document.createElement('canvas')
Expand All @@ -37,9 +38,10 @@ export default async function getCroppedImg(
canvas.width = safeArea
canvas.height = safeArea

// translate canvas context to a central location to allow rotating around the center.
// translate canvas context to a central location to allow rotating and flipping around the center.
ctx.translate(safeArea / 2, safeArea / 2)
ctx.rotate(getRadianAngle(rotation))
ctx.scale(flip.horizontal ? -1 : 1, flip.vertical ? -1 : 1)
ctx.translate(-safeArea / 2, -safeArea / 2)

// draw rotated image and store data.
Expand Down
34 changes: 32 additions & 2 deletions docs/src/components/Demo/index.tsx
Original file line number Diff line number Diff line change
@@ -1,8 +1,10 @@
import Button from '@material-ui/core/Button'
import IconButton from '@material-ui/core/IconButton'
import NoSsr from '@material-ui/core/NoSsr'
import Slider from '@material-ui/core/Slider'
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
import Typography from '@material-ui/core/Typography'
import FlipIcon from '@material-ui/icons/Flip'
import React, { useCallback, useState } from 'react'
import Cropper from 'react-easy-crop'
import { Area } from 'react-easy-crop/types'
Expand Down Expand Up @@ -61,6 +63,7 @@ const Demo: React.FC = props => {
const classes = useStyles(props)
const [crop, setCrop] = useState({ x: 0, y: 0 })
const [rotation, setRotation] = useState(0)
const [flip, setFlip] = useState({ horizontal: false, vertical: false })
const [zoom, setZoom] = useState(1)
const [croppedAreaPixels, setCroppedAreaPixels] = useState<Area | null>(null)
const [croppedImage, setCroppedImage] = useState<string | null>(null)
Expand All @@ -71,12 +74,12 @@ const Demo: React.FC = props => {

const showCroppedImage = useCallback(async () => {
try {
const croppedImage = await getCroppedImg(dogImg, croppedAreaPixels, rotation)
const croppedImage = await getCroppedImg(dogImg, croppedAreaPixels, rotation, flip)
setCroppedImage(croppedImage)
} catch (e) {
console.error(e)
}
}, [croppedAreaPixels, rotation])
}, [croppedAreaPixels, rotation, flip])

const onClose = useCallback(() => {
setCroppedImage(null)
Expand All @@ -88,6 +91,13 @@ const Demo: React.FC = props => {
<NoSsr>
<Cropper
image={dogImg}
transform={[
`translate(${crop.x}px, ${crop.y}px)`,
`rotateZ(${rotation}deg)`,
`rotateY(${flip.horizontal ? 180 : 0}deg)`,
`rotateX(${flip.vertical ? 180 : 0}deg)`,
`scale(${zoom})`,
].join(' ')}
crop={crop}
rotation={rotation}
zoom={zoom}
Expand Down Expand Up @@ -128,6 +138,26 @@ const Demo: React.FC = props => {
onChange={(e, rotation) => setRotation(rotation as number)}
/>
</div>
<div className={classes.sliderContainer}>
<IconButton
aria-label="Flip Horizontal"
onClick={() => {
setFlip(prev => ({ horizontal: !prev.horizontal, vertical: prev.vertical }))
setRotation(prev => 360 - prev)
}}
>
<FlipIcon />
</IconButton>
<IconButton
aria-label="Flip Vertical"
onClick={() => {
setFlip(prev => ({ horizontal: prev.horizontal, vertical: !prev.vertical }))
setRotation(prev => 360 - prev)
}}
>
<FlipIcon style={{ transform: 'rotate(90deg)' }} />
</IconButton>
</div>
<Button
onClick={showCroppedImage}
variant="contained"
Expand Down
68 changes: 59 additions & 9 deletions examples/src/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ type State = {
imageSrc: string
crop: Point
rotation: number
flip: { horizontal: boolean; vertical: boolean }
zoom: number
aspect: number
cropShape: 'rect' | 'round'
Expand All @@ -25,6 +26,7 @@ class App extends React.Component<{}, State> {
imageSrc,
crop: { x: 0, y: 0 },
rotation: 0,
flip: { horizontal: false, vertical: false },
zoom: 1,
aspect: 4 / 3,
cropShape: 'rect',
Expand Down Expand Up @@ -60,15 +62,56 @@ class App extends React.Component<{}, State> {
render() {
return (
<div className="App">
<input
type="range"
min={0}
max={360}
onChange={({ target: { value: rotation } }) =>
this.setState({ rotation: Number(rotation) })
}
style={{ position: 'fixed', zIndex: 9999999 }}
/>
<div className="controls">
<div>
<label>
Rotation
<input
type="range"
min={0}
max={360}
value={this.state.rotation}
onChange={({ target: { value: rotation } }) =>
this.setState({ rotation: Number(rotation) })
}
/>
</label>
</div>
<div>
<label>
<input
type="checkbox"
checked={this.state.flip.horizontal}
onChange={() =>
this.setState(prev => ({
rotation: 360 - prev.rotation,
flip: {
horizontal: !prev.flip.horizontal,
vertical: prev.flip.vertical,
},
}))
}
/>
Flip Horizontal
</label>
<label>
<input
type="checkbox"
checked={this.state.flip.vertical}
onChange={() =>
this.setState(prev => ({
rotation: 360 - prev.rotation,
flip: {
horizontal: prev.flip.horizontal,
vertical: !prev.flip.vertical,
},
}))
}
/>
Flip Vertical
</label>
</div>
</div>
<div className="crop-container">
<Cropper
image={this.state.imageSrc}
Expand All @@ -89,6 +132,13 @@ class App extends React.Component<{}, State> {
initialCroppedAreaPixels={
!!urlArgs.setInitialCrop ? { width: 699, height: 524, x: 875, y: 157 } : undefined // used to set the initial crop in e2e test
}
transform={[
`translate(${this.state.crop.x}px, ${this.state.crop.y}px)`,
`rotateZ(${this.state.rotation}deg)`,
`rotateY(${this.state.flip.horizontal ? 180 : 0}deg)`,
`rotateX(${this.state.flip.vertical ? 180 : 0}deg)`,
`scale(${this.state.zoom})`,
].join(' ')}
/>
</div>
</div>
Expand Down
5 changes: 5 additions & 0 deletions examples/src/styles.css
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,11 @@ body {
bottom: 0;
}

.controls {
z-index: 1;
position: fixed;
}

.crop-container {
position: absolute;
top: 0;
Expand Down
8 changes: 6 additions & 2 deletions src/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ import cssStyles from './styles.css'
type Props = {
image?: string
video?: string
transform?: string
crop: Point
zoom: number
rotation: number
Expand Down Expand Up @@ -439,6 +440,7 @@ class Cropper extends React.Component<Props, State> {
image,
video,
mediaProps,
transform,
crop: { x, y },
rotation,
zoom,
Expand Down Expand Up @@ -466,7 +468,8 @@ class Cropper extends React.Component<Props, State> {
ref={(el: HTMLImageElement) => (this.imageRef = el)}
style={{
...mediaStyle,
transform: `translate(${x}px, ${y}px) rotate(${rotation}deg) scale(${zoom})`,
transform:
transform || `translate(${x}px, ${y}px) rotate(${rotation}deg) scale(${zoom})`,
}}
onLoad={this.onMediaLoad}
/>
Expand All @@ -483,7 +486,8 @@ class Cropper extends React.Component<Props, State> {
onLoadedMetadata={this.onMediaLoad}
style={{
...mediaStyle,
transform: `translate(${x}px, ${y}px) rotate(${rotation}deg) scale(${zoom})`,
transform:
transform || `translate(${x}px, ${y}px) rotate(${rotation}deg) scale(${zoom})`,
}}
controls={false}
/>
Expand Down