import { uniqueId, isMobile, getFromCache } from 'utils/utils' import vec from 'utils/vec' import TextAreaUtils from 'utils/text-area' import { TextShape, ShapeType } from 'types' import { defaultStyle, getFontSize, getFontStyle, getShapeStyle, } from 'state/shape-styles' import styled from 'styles' import state from 'state' import { registerShapeUtils } from './register' // A div used for measurement document.getElementById('__textMeasure')?.remove() const mdiv = document.createElement('pre') mdiv.id = '__textMeasure' Object.assign(mdiv.style, { whiteSpace: 'pre', width: 'auto', border: '1px solid red', padding: '4px', margin: '0px', opacity: '0', position: 'absolute', top: '-500px', left: '0px', zIndex: '9999', pointerEvents: 'none', userSelect: 'none', alignmentBaseline: 'mathematical', dominantBaseline: 'mathematical', }) mdiv.tabIndex = -1 document.body.appendChild(mdiv) function normalizeText(text: string) { return text.replace(/\r?\n|\r/g, '\n') } const text = registerShapeUtils({ isForeignObject: true, canChangeAspectRatio: false, canEdit: true, boundsCache: new WeakMap([]), defaultProps: { id: uniqueId(), type: ShapeType.Text, name: 'Text', parentId: 'page1', childIndex: 0, point: [0, 0], rotation: 0, style: defaultStyle, text: '', scale: 1, }, shouldRender(shape, prev) { return ( shape.text !== prev.text || shape.scale !== prev.scale || shape.style !== prev.style ) }, render(shape, { isEditing, ref }) { const { id, text, style } = shape const styles = getShapeStyle(style) const font = getFontStyle(shape.scale, shape.style) const bounds = this.getBounds(shape) function handleChange(e: React.ChangeEvent) { state.send('EDITED_SHAPE', { id, change: { text: normalizeText(e.currentTarget.value) }, }) } function handleKeyDown(e: React.KeyboardEvent) { if (e.key === 'Escape') return e.stopPropagation() if (e.key === 'Tab') { e.preventDefault() if (e.shiftKey) { TextAreaUtils.unindent(e.currentTarget) } else { TextAreaUtils.indent(e.currentTarget) } state.send('EDITED_SHAPE', { id, change: { text: normalizeText(e.currentTarget.value), }, }) } } function handleBlur() { setTimeout(() => state.send('BLURRED_EDITING_SHAPE', { id }), 0) } function handleFocus(e: React.FocusEvent) { e.currentTarget.select() state.send('FOCUSED_EDITING_SHAPE', { id }) } function handlePointerDown(e: React.PointerEvent) { if (e.currentTarget.selectionEnd !== 0) { e.currentTarget.selectionEnd = 0 } } const fontSize = getFontSize(shape.style.size) * shape.scale const lineHeight = fontSize * 1.4 if (ref === undefined) { throw Error('This component should receive a ref.') } if (!isEditing) { return ( {text.split('\n').map((str, i) => ( {str} ))} ) } return ( } style={{ font, color: styles.stroke, }} name="text" defaultValue={text} tabIndex={-1} autoComplete="false" autoCapitalize="false" autoCorrect="false" autoSave="false" placeholder="" color={styles.stroke} autoFocus={!!isMobile()} onFocus={handleFocus} onBlur={handleBlur} onKeyDown={handleKeyDown} onChange={handleChange} onPointerDown={handlePointerDown} /> ) }, getBounds(shape) { const bounds = getFromCache(this.boundsCache, shape, (cache) => { mdiv.innerHTML = `${shape.text}‍` mdiv.style.font = getFontStyle(shape.scale, shape.style) const [minX, minY] = shape.point const [width, height] = [mdiv.offsetWidth, mdiv.offsetHeight] cache.set(shape, { minX, maxX: minX + width, minY, maxY: minY + height, width, height, }) }) return bounds }, hitTest() { return true }, transform(shape, bounds, { initialShape, scaleX, scaleY }) { if (shape.rotation === 0 && !shape.isAspectRatioLocked) { shape.point = [bounds.minX, bounds.minY] shape.scale = initialShape.scale * Math.abs(scaleX) } else { shape.point = [bounds.minX, bounds.minY] shape.rotation = (scaleX < 0 && scaleY >= 0) || (scaleY < 0 && scaleX >= 0) ? -initialShape.rotation : initialShape.rotation shape.scale = initialShape.scale * Math.abs(Math.min(scaleX, scaleY)) } return this }, transformSingle(shape, bounds, { initialShape, scaleX }) { shape.point = [bounds.minX, bounds.minY] shape.scale = initialShape.scale * Math.abs(scaleX) return this }, onBoundsReset(shape) { const center = this.getCenter(shape) this.boundsCache.delete(shape) shape.scale = 1 const newCenter = this.getCenter(shape) shape.point = vec.add(shape.point, vec.sub(center, newCenter)) return this }, applyStyles(shape, style) { const center = this.getCenter(shape) this.boundsCache.delete(shape) Object.assign(shape.style, style) const newCenter = this.getCenter(shape) shape.point = vec.add(shape.point, vec.sub(center, newCenter)) return this }, shouldDelete(shape) { return shape.text.length === 0 }, }) export default text const StyledTextArea = styled('textarea', { zIndex: 1, width: '100%', height: '100%', border: 'none', padding: '4px', whiteSpace: 'pre', alignmentBaseline: 'mathematical', dominantBaseline: 'mathematical', resize: 'none', minHeight: 1, minWidth: 1, lineHeight: 1.4, outline: 0, backgroundColor: '$boundsBg', overflow: 'hidden', pointerEvents: 'all', backfaceVisibility: 'hidden', display: 'inline-block', userSelect: 'text', WebkitUserSelect: 'text', WebkitTouchCallout: 'none', })