tldraw
Version:
A tiny little drawing editor.
91 lines (75 loc) • 2.04 kB
text/typescript
import { RotateCorner, StateNode, TLPointerEventInfo } from '@tldraw/editor'
import { CursorTypeMap } from './PointingResizeHandle'
type PointingRotateHandleInfo = Extract<TLPointerEventInfo, { target: 'selection' }> & {
onInteractionEnd?: string | (() => void)
}
export class PointingRotateHandle extends StateNode {
static override id = 'pointing_rotate_handle'
private info = {} as PointingRotateHandleInfo
private updateCursor() {
this.editor.setCursor({
type: CursorTypeMap[this.info.handle as RotateCorner],
rotation: this.editor.getSelectionRotation(),
})
}
override onEnter(info: PointingRotateHandleInfo) {
this.info = info
if (typeof info.onInteractionEnd === 'string') {
this.parent.setCurrentToolIdMask(info.onInteractionEnd)
}
this.updateCursor()
}
override onExit() {
this.parent.setCurrentToolIdMask(undefined)
this.editor.setCursor({ type: 'default', rotation: 0 })
}
override onPointerMove() {
if (this.editor.inputs.getIsDragging()) {
this.startRotating()
}
}
override onLongPress() {
this.startRotating()
}
private startRotating() {
if (this.editor.getIsReadonly()) return
this.parent.transition('rotating', this.info)
}
override onPointerUp() {
this.complete()
}
override onCancel() {
this.cancel()
}
override onComplete() {
this.cancel()
}
override onInterrupt() {
this.cancel()
}
private complete() {
const { onInteractionEnd } = this.info
if (onInteractionEnd) {
if (typeof onInteractionEnd === 'string') {
// Return to the tool that was active before this one, whether tool lock is turned on or not!
this.editor.setCurrentTool(onInteractionEnd, {})
} else {
onInteractionEnd?.()
}
return
}
this.parent.transition('idle')
}
private cancel() {
const { onInteractionEnd } = this.info
if (onInteractionEnd) {
if (typeof onInteractionEnd === 'string') {
this.editor.setCurrentTool(onInteractionEnd, {})
} else {
onInteractionEnd()
}
return
}
this.parent.transition('idle')
}
}