Tldraw/packages/tldraw/src/lib/shapes/image/ImageShapeUtil.tsx

316 wiersze
8.7 KiB
TypeScript

/* eslint-disable react-hooks/rules-of-hooks */
import {
BaseBoxShapeUtil,
FileHelpers,
HTMLContainer,
TLImageShape,
TLOnDoubleClickHandler,
TLShapePartial,
Vec,
imageShapeMigrations,
imageShapeProps,
structuredClone,
toDomPrecision,
} from '@tldraw/editor'
import { useEffect, useState } from 'react'
import { BrokenAssetIcon } from '../shared/BrokenAssetIcon'
import { HyperlinkButton } from '../shared/HyperlinkButton'
import { usePrefersReducedMotion } from '../shared/usePrefersReducedMotion'
async function getDataURIFromURL(url: string): Promise<string> {
const response = await fetch(url)
const blob = await response.blob()
return FileHelpers.blobToDataUrl(blob)
}
/** @public */
export class ImageShapeUtil extends BaseBoxShapeUtil<TLImageShape> {
static override type = 'image' as const
static override props = imageShapeProps
static override migrations = imageShapeMigrations
override isAspectRatioLocked = () => true
override canCrop = () => true
override getDefaultProps(): TLImageShape['props'] {
return {
w: 100,
h: 100,
assetId: null,
playing: true,
url: '',
crop: null,
}
}
component(shape: TLImageShape) {
const isCropping = this.editor.getCroppingShapeId() === shape.id
const prefersReducedMotion = usePrefersReducedMotion()
const [staticFrameSrc, setStaticFrameSrc] = useState('')
const asset = shape.props.assetId ? this.editor.getAsset(shape.props.assetId) : undefined
const isSelected = shape.id === this.editor.getOnlySelectedShape()?.id
useEffect(() => {
if (asset?.props.src && 'mimeType' in asset.props && asset?.props.mimeType === 'image/gif') {
let cancelled = false
const url = asset.props.src
if (!url) return
const image = new Image()
image.onload = () => {
if (cancelled) return
const canvas = document.createElement('canvas')
canvas.width = image.width
canvas.height = image.height
const ctx = canvas.getContext('2d')
if (!ctx) return
ctx.drawImage(image, 0, 0)
setStaticFrameSrc(canvas.toDataURL())
}
image.crossOrigin = 'anonymous'
image.src = url
return () => {
cancelled = true
}
}
}, [prefersReducedMotion, asset?.props])
if (asset?.type === 'bookmark') {
throw Error("Bookmark assets can't be rendered as images")
}
const showCropPreview =
isSelected &&
isCropping &&
this.editor.isInAny('select.crop', 'select.cropping', 'select.pointing_crop_handle')
// We only want to reduce motion for mimeTypes that have motion
const reduceMotion =
prefersReducedMotion &&
(asset?.props.mimeType?.includes('video') || asset?.props.mimeType?.includes('gif'))
const containerStyle = getCroppedContainerStyle(shape)
if (!asset?.props.src) {
return (
<HTMLContainer
id={shape.id}
style={{
overflow: 'hidden',
width: shape.props.w,
height: shape.props.h,
color: 'var(--color-text-3)',
backgroundColor: asset ? 'transparent' : 'var(--color-low)',
border: asset ? 'none' : '1px solid var(--color-low-border)',
}}
>
<div className="tl-image-container" style={containerStyle}>
{asset ? null : <BrokenAssetIcon />}
</div>
)
{'url' in shape.props && shape.props.url && (
<HyperlinkButton url={shape.props.url} zoomLevel={this.editor.getZoomLevel()} />
)}
</HTMLContainer>
)
}
return (
<>
{showCropPreview && (
<div style={containerStyle}>
<div
className="tl-image"
style={{
opacity: 0.1,
backgroundImage: `url(${
!shape.props.playing || reduceMotion ? staticFrameSrc : asset.props.src
})`,
}}
draggable={false}
/>
</div>
)}
<HTMLContainer
id={shape.id}
style={{ overflow: 'hidden', width: shape.props.w, height: shape.props.h }}
>
<div className="tl-image-container" style={containerStyle}>
<div
className="tl-image"
style={{
backgroundImage: `url(${
!shape.props.playing || reduceMotion ? staticFrameSrc : asset.props.src
})`,
}}
draggable={false}
/>
{asset.props.isAnimated && !shape.props.playing && (
<div className="tl-image__tg">GIF</div>
)}
</div>
)
{shape.props.url && (
<HyperlinkButton url={shape.props.url} zoomLevel={this.editor.getZoomLevel()} />
)}
</HTMLContainer>
</>
)
}
indicator(shape: TLImageShape) {
const isCropping = this.editor.getCroppingShapeId() === shape.id
if (isCropping) return null
return <rect width={toDomPrecision(shape.props.w)} height={toDomPrecision(shape.props.h)} />
}
override async toSvg(shape: TLImageShape) {
const g = document.createElementNS('http://www.w3.org/2000/svg', 'g')
const asset = shape.props.assetId ? this.editor.getAsset(shape.props.assetId) : null
if (!asset) return g
let src = asset?.props.src || ''
if (src.startsWith('http') || src.startsWith('/') || src.startsWith('./')) {
// If it's a remote image, we need to fetch it and convert it to a data URI
src = (await getDataURIFromURL(src)) || ''
}
const image = document.createElementNS('http://www.w3.org/2000/svg', 'image')
image.setAttributeNS('http://www.w3.org/1999/xlink', 'href', src)
const containerStyle = getCroppedContainerStyle(shape)
const crop = shape.props.crop
if (containerStyle.transform && crop) {
const { transform, width, height } = containerStyle
const croppedWidth = (crop.bottomRight.x - crop.topLeft.x) * width
const croppedHeight = (crop.bottomRight.y - crop.topLeft.y) * height
const points = [
new Vec(0, 0),
new Vec(croppedWidth, 0),
new Vec(croppedWidth, croppedHeight),
new Vec(0, croppedHeight),
]
const polygon = document.createElementNS('http://www.w3.org/2000/svg', 'polygon')
polygon.setAttribute('points', points.map((p) => `${p.x},${p.y}`).join(' '))
const clipPath = document.createElementNS('http://www.w3.org/2000/svg', 'clipPath')
clipPath.setAttribute('id', 'cropClipPath')
clipPath.appendChild(polygon)
const defs = document.createElementNS('http://www.w3.org/2000/svg', 'defs')
defs.appendChild(clipPath)
g.appendChild(defs)
const innerElement = document.createElementNS('http://www.w3.org/2000/svg', 'g')
innerElement.setAttribute('clip-path', 'url(#cropClipPath)')
image.setAttribute('width', width.toString())
image.setAttribute('height', height.toString())
image.style.transform = transform
innerElement.appendChild(image)
g.appendChild(innerElement)
} else {
image.setAttribute('width', shape.props.w.toString())
image.setAttribute('height', shape.props.h.toString())
g.appendChild(image)
}
return g
}
override onDoubleClick = (shape: TLImageShape) => {
const asset = shape.props.assetId ? this.editor.getAsset(shape.props.assetId) : undefined
if (!asset) return
const canPlay =
asset.props.src && 'mimeType' in asset.props && asset.props.mimeType === 'image/gif'
if (!canPlay) return
this.editor.updateShapes([
{
type: 'image',
id: shape.id,
props: {
playing: !shape.props.playing,
},
},
])
}
override onDoubleClickEdge: TLOnDoubleClickHandler<TLImageShape> = (shape) => {
const props = shape.props
if (!props) return
if (this.editor.getCroppingShapeId() !== shape.id) {
return
}
const crop = structuredClone(props.crop) || {
topLeft: { x: 0, y: 0 },
bottomRight: { x: 1, y: 1 },
}
// The true asset dimensions
const w = (1 / (crop.bottomRight.x - crop.topLeft.x)) * shape.props.w
const h = (1 / (crop.bottomRight.y - crop.topLeft.y)) * shape.props.h
const pointDelta = new Vec(crop.topLeft.x * w, crop.topLeft.y * h).rot(shape.rotation)
const partial: TLShapePartial<TLImageShape> = {
id: shape.id,
type: shape.type,
x: shape.x - pointDelta.x,
y: shape.y - pointDelta.y,
props: {
crop: {
topLeft: { x: 0, y: 0 },
bottomRight: { x: 1, y: 1 },
},
w,
h,
},
}
this.editor.updateShapes([partial])
}
}
/**
* When an image is cropped we need to translate the image to show the portion withing the cropped
* area. We do this by translating the image by the negative of the top left corner of the crop
* area.
*
* @param shape - Shape The image shape for which to get the container style
* @returns - Styles to apply to the image container
*/
function getCroppedContainerStyle(shape: TLImageShape) {
const crop = shape.props.crop
const topLeft = crop?.topLeft
if (!topLeft) {
return {
width: shape.props.w,
height: shape.props.h,
}
}
const w = (1 / (crop.bottomRight.x - crop.topLeft.x)) * shape.props.w
const h = (1 / (crop.bottomRight.y - crop.topLeft.y)) * shape.props.h
const offsetX = -topLeft.x * w
const offsetY = -topLeft.y * h
return {
transform: `translate(${offsetX}px, ${offsetY}px)`,
width: w,
height: h,
}
}