kopia lustrzana https://github.com/Tldraw/Tldraw
264 wiersze
7.6 KiB
TypeScript
264 wiersze
7.6 KiB
TypeScript
import { TLDrawShape, TLHighlightShape } from '@tldraw/tlschema'
|
|
import { last } from '@tldraw/utils'
|
|
import { TestApp } from '../TestApp'
|
|
|
|
jest.useFakeTimers()
|
|
|
|
let app: TestApp
|
|
|
|
afterEach(() => {
|
|
app?.dispose()
|
|
})
|
|
|
|
beforeEach(() => {
|
|
app = new TestApp()
|
|
|
|
app.createShapes([])
|
|
})
|
|
|
|
type DrawableShape = TLDrawShape | TLHighlightShape
|
|
|
|
for (const toolType of ['draw', 'highlight'] as const) {
|
|
describe(`When ${toolType}ing...`, () => {
|
|
it('Creates a dot', () => {
|
|
app
|
|
.setSelectedTool(toolType)
|
|
.pointerDown(60, 60)
|
|
.expectToBeIn(`${toolType}.drawing`)
|
|
.pointerUp()
|
|
.expectToBeIn(`${toolType}.idle`)
|
|
|
|
expect(app.shapesArray).toHaveLength(1)
|
|
|
|
const shape = app.shapesArray[0] as DrawableShape
|
|
expect(shape.type).toBe(toolType)
|
|
expect(shape.props.segments.length).toBe(1)
|
|
|
|
const segment = shape.props.segments[0]
|
|
expect(segment.type).toBe('free')
|
|
})
|
|
|
|
it('Creates a dot when shift is held down', () => {
|
|
app
|
|
.setSelectedTool(toolType)
|
|
.keyDown('Shift')
|
|
.pointerDown(60, 60)
|
|
.expectToBeIn(`${toolType}.drawing`)
|
|
.pointerUp()
|
|
.expectToBeIn(`${toolType}.idle`)
|
|
|
|
expect(app.shapesArray).toHaveLength(1)
|
|
|
|
const shape = app.shapesArray[0] as DrawableShape
|
|
expect(shape.type).toBe(toolType)
|
|
expect(shape.props.segments.length).toBe(1)
|
|
|
|
const segment = shape.props.segments[0]
|
|
expect(segment.type).toBe('straight')
|
|
})
|
|
|
|
it('Creates a free draw line when shift is not held', () => {
|
|
app.setSelectedTool(toolType).pointerDown(10, 10).pointerMove(20, 20)
|
|
|
|
const shape = app.shapesArray[0] as DrawableShape
|
|
expect(shape.props.segments.length).toBe(1)
|
|
|
|
const segment = shape.props.segments[0]
|
|
expect(segment.type).toBe('free')
|
|
})
|
|
|
|
it('Creates a straight line when shift is held', () => {
|
|
app.setSelectedTool(toolType).keyDown('Shift').pointerDown(10, 10).pointerMove(20, 20)
|
|
|
|
const shape = app.shapesArray[0] as DrawableShape
|
|
expect(shape.props.segments.length).toBe(1)
|
|
|
|
const segment = shape.props.segments[0]
|
|
expect(segment.type).toBe('straight')
|
|
|
|
const points = segment.points
|
|
expect(points.length).toBe(2)
|
|
})
|
|
|
|
it('Switches between segment types when shift is pressed / released (starting with shift up)', () => {
|
|
app
|
|
.setSelectedTool(toolType)
|
|
.pointerDown(10, 10)
|
|
.pointerMove(20, 20)
|
|
.keyDown('Shift')
|
|
.pointerMove(30, 30)
|
|
.keyUp('Shift')
|
|
.pointerMove(40, 40)
|
|
.pointerUp()
|
|
|
|
const shape = app.shapesArray[0] as DrawableShape
|
|
expect(shape.props.segments.length).toBe(3)
|
|
|
|
expect(shape.props.segments[0].type).toBe('free')
|
|
expect(shape.props.segments[1].type).toBe('straight')
|
|
expect(shape.props.segments[2].type).toBe('free')
|
|
})
|
|
|
|
it('Switches between segment types when shift is pressed / released (starting with shift down)', () => {
|
|
app
|
|
.setSelectedTool(toolType)
|
|
.keyDown('Shift')
|
|
.pointerDown(10, 10)
|
|
.pointerMove(20, 20)
|
|
.keyUp('Shift')
|
|
.pointerMove(30, 30)
|
|
.keyDown('Shift')
|
|
.pointerMove(40, 40)
|
|
.pointerUp()
|
|
|
|
const shape = app.shapesArray[0] as DrawableShape
|
|
expect(shape.props.segments.length).toBe(3)
|
|
|
|
expect(shape.props.segments[0].type).toBe('straight')
|
|
expect(shape.props.segments[1].type).toBe('free')
|
|
expect(shape.props.segments[2].type).toBe('straight')
|
|
})
|
|
|
|
it('Extends previously drawn line when shift is held', () => {
|
|
app
|
|
.setSelectedTool(toolType)
|
|
.keyDown('Shift')
|
|
.pointerDown(10, 10)
|
|
.pointerUp()
|
|
.pointerDown(20, 20)
|
|
|
|
const shape1 = app.shapesArray[0] as DrawableShape
|
|
expect(shape1.props.segments.length).toBe(2)
|
|
expect(shape1.props.segments[0].type).toBe('straight')
|
|
expect(shape1.props.segments[1].type).toBe('straight')
|
|
|
|
app.pointerUp().pointerDown(30, 30).pointerUp()
|
|
|
|
const shape2 = app.shapesArray[0] as DrawableShape
|
|
expect(shape2.props.segments.length).toBe(3)
|
|
expect(shape2.props.segments[2].type).toBe('straight')
|
|
})
|
|
|
|
it('Does not extends previously drawn line after switching to another tool', () => {
|
|
app
|
|
.setSelectedTool(toolType)
|
|
.pointerDown(10, 10)
|
|
.pointerUp()
|
|
.setSelectedTool('select')
|
|
.setSelectedTool(toolType)
|
|
.keyDown('Shift')
|
|
.pointerDown(20, 20)
|
|
.pointerMove(30, 30)
|
|
|
|
expect(app.shapesArray).toHaveLength(2)
|
|
|
|
const shape1 = app.shapesArray[0] as DrawableShape
|
|
expect(shape1.props.segments.length).toBe(1)
|
|
expect(shape1.props.segments[0].type).toBe('free')
|
|
|
|
const shape2 = app.shapesArray[1] as DrawableShape
|
|
expect(shape2.props.segments.length).toBe(1)
|
|
expect(shape2.props.segments[0].type).toBe('straight')
|
|
})
|
|
|
|
it('Snaps to 15 degree angle when shift is held', () => {
|
|
const magnitude = 10
|
|
const angle = (17 * Math.PI) / 180
|
|
const x = magnitude * Math.cos(angle)
|
|
const y = magnitude * Math.sin(angle)
|
|
|
|
const snappedAngle = (15 * Math.PI) / 180
|
|
const snappedX = magnitude * Math.cos(snappedAngle)
|
|
const snappedY = magnitude * Math.sin(snappedAngle)
|
|
|
|
app.setSelectedTool(toolType).keyDown('Shift').pointerDown(0, 0).pointerMove(x, y)
|
|
|
|
const shape = app.shapesArray[0] as DrawableShape
|
|
const segment = shape.props.segments[0]
|
|
expect(segment.points[1].x).toBeCloseTo(snappedX)
|
|
expect(segment.points[1].y).toBeCloseTo(snappedY)
|
|
})
|
|
|
|
it('Doesnt snap to 15 degree angle when cmd is held', () => {
|
|
const magnitude = 10
|
|
const angle = (17 * Math.PI) / 180
|
|
const x = magnitude * Math.cos(angle)
|
|
const y = magnitude * Math.sin(angle)
|
|
|
|
app.setSelectedTool(toolType).keyDown('Meta').pointerDown(0, 0).pointerMove(x, y)
|
|
|
|
const shape = app.shapesArray[0] as DrawableShape
|
|
const segment = shape.props.segments[0]
|
|
expect(segment.points[1].x).toBeCloseTo(x)
|
|
expect(segment.points[1].y).toBeCloseTo(y)
|
|
})
|
|
|
|
it('Snaps to start or end of straight segments in self when shift + cmd is held', () => {
|
|
app
|
|
.setSelectedTool(toolType)
|
|
.keyDown('Shift')
|
|
.pointerDown(0, 0)
|
|
.pointerUp()
|
|
.pointerDown(0, 10)
|
|
.pointerUp()
|
|
.pointerDown(10, 0)
|
|
.pointerUp()
|
|
.pointerDown(10, 0)
|
|
.pointerMove(1, 0)
|
|
|
|
const shape1 = app.shapesArray[0] as DrawableShape
|
|
const segment1 = last(shape1.props.segments)!
|
|
const point1 = last(segment1.points)!
|
|
expect(point1.x).toBe(1)
|
|
|
|
app.keyDown('Meta')
|
|
const shape2 = app.shapesArray[0] as DrawableShape
|
|
const segment2 = last(shape2.props.segments)!
|
|
const point2 = last(segment2.points)!
|
|
expect(point2.x).toBe(0)
|
|
})
|
|
|
|
it('Snaps to position along straight segments in self when shift + cmd is held', () => {
|
|
app
|
|
.setSelectedTool(toolType)
|
|
.keyDown('Shift')
|
|
.pointerDown(0, 0)
|
|
.pointerUp()
|
|
.pointerDown(0, 10)
|
|
.pointerUp()
|
|
.pointerDown(10, 5)
|
|
.pointerUp()
|
|
.pointerDown(10, 5)
|
|
.pointerMove(1, 5)
|
|
|
|
const shape1 = app.shapesArray[0] as DrawableShape
|
|
const segment1 = last(shape1.props.segments)!
|
|
const point1 = last(segment1.points)!
|
|
expect(point1.x).toBe(1)
|
|
|
|
app.keyDown('Meta')
|
|
const shape2 = app.shapesArray[0] as DrawableShape
|
|
const segment2 = last(shape2.props.segments)!
|
|
const point2 = last(segment2.points)!
|
|
expect(point2.x).toBe(0)
|
|
})
|
|
|
|
it('Deletes very short lines on interrupt', () => {
|
|
app.setSelectedTool(toolType).pointerDown(0, 0).pointerMove(0.1, 0.1).interrupt()
|
|
expect(app.shapesArray).toHaveLength(0)
|
|
})
|
|
|
|
it('Does not delete longer lines on interrupt', () => {
|
|
app.setSelectedTool(toolType).pointerDown(0, 0).pointerMove(5, 5).interrupt()
|
|
expect(app.shapesArray).toHaveLength(1)
|
|
})
|
|
|
|
it('Completes on cancel', () => {
|
|
app.setSelectedTool(toolType).pointerDown(0, 0).pointerMove(5, 5).cancel()
|
|
expect(app.shapesArray).toHaveLength(1)
|
|
const shape = app.shapesArray[0] as DrawableShape
|
|
expect(shape.props.segments.length).toBe(1)
|
|
})
|
|
})
|
|
}
|