import { TestEditor } from '../../../test/TestEditor'
import { GeoShapeTool } from './GeoShapeTool'

let editor: TestEditor

beforeEach(() => {
	editor = new TestEditor()
})
afterEach(() => {
	editor?.dispose()
})

describe(GeoShapeTool, () => {
	it('Creates geo shapes on click-and-drag, supports undo and redo', () => {
		expect(editor.getCurrentPageShapes().length).toBe(0)

		editor.setCurrentTool('geo')
		editor.pointerDown(50, 50)
		editor.pointerMove(100, 100)
		editor.pointerUp()

		expect(editor.getCurrentPageShapes().length).toBe(1)
		expect(editor.getCurrentPageShapes()[0]?.type).toBe('geo')
		expect(editor.getSelectedShapeIds()[0]).toBe(editor.getCurrentPageShapes()[0]?.id)

		editor.undo()

		expect(editor.getCurrentPageShapes().length).toBe(0)

		editor.redo()

		expect(editor.getCurrentPageShapes().length).toBe(1)
	})

	it('Creates geo shapes on click, supports undo and redo', () => {
		expect(editor.getCurrentPageShapes().length).toBe(0)

		editor.setCurrentTool('geo')
		editor.pointerDown(50, 50)
		editor.pointerUp(50, 50)

		expect(editor.getCurrentPageShapes().length).toBe(1)
		expect(editor.getCurrentPageShapes()[0]?.type).toBe('geo')
		expect(editor.getSelectedShapeIds()[0]).toBe(editor.getCurrentPageShapes()[0]?.id)

		editor.undo()

		expect(editor.getCurrentPageShapes().length).toBe(0)

		editor.redo()

		expect(editor.getCurrentPageShapes().length).toBe(1)
	})

	it('Creates geo shapes when scaled', () => {
		editor.setCurrentTool('geo')
		editor.pointerMove(50, 50)
		editor.pointerDown()
		editor.pointerMove(100, 100)
		editor.pointerUp()

		expect(editor.getLastCreatedShape()).toMatchObject({
			props: {
				w: 50,
				h: 50,
				scale: 1,
			},
		})

		editor.user.updateUserPreferences({ isDynamicSizeMode: true })
		editor.zoomIn() // 1 -> 2

		editor.setCurrentTool('geo')
		editor.pointerMove(50, 50)
		editor.pointerDown()
		editor.pointerMove(100, 100)
		editor.pointerUp()

		expect(editor.getLastCreatedShape()).toMatchObject({
			props: {
				w: 25,
				h: 25,
				scale: 0.5,
			},
		})

		editor.zoomOut().zoomOut() // 2 -> 1 -> .5

		editor.setCurrentTool('geo')
		editor.pointerMove(50, 50)
		editor.pointerDown()
		editor.pointerMove(100, 100)
		editor.pointerUp()

		expect(editor.getLastCreatedShape()).toMatchObject({
			props: {
				w: 100,
				h: 100,
				scale: 2,
			},
		})
	})
})

describe('When selecting the tool', () => {
	it('selects the tool and enters the idle state', () => {
		editor.setCurrentTool('geo')
		editor.expectToBeIn('geo.idle')
	})
})

describe('When in the idle state', () => {
	it('Enters pointing state on pointer down', () => {
		editor.setCurrentTool('geo')
		editor.pointerDown(100, 100)
		editor.expectToBeIn('geo.pointing')
	})

	it('Switches back to select tool on cancel', () => {
		editor.setCurrentTool('geo')
		editor.cancel()
		editor.expectToBeIn('select.idle')
	})

	it('Does nothing on interrupt', () => {
		editor.setCurrentTool('geo')
		editor.interrupt()
		editor.expectToBeIn('geo.idle')
	})

	it('Enters edit shape state on "Enter" key up when we have one geo shape', () => {
		editor.setCurrentTool('geo')
		editor.pointerDown(50, 50)
		editor.pointerMove(100, 100)
		editor.pointerUp(100, 100)

		editor.keyPress('Enter')
		editor.expectToBeIn('select.editing_shape')
	})

	it('Does not enter edit shape state on "Enter" key up when multiple geo shapes are selected', () => {
		editor.setCurrentTool('geo')
		editor.pointerDown(50, 50)
		editor.pointerMove(100, 100)
		editor.pointerUp(100, 100)

		editor.setCurrentTool('geo')
		editor.pointerDown(150, 150)
		editor.pointerMove(200, 200)
		editor.pointerUp(200, 200)

		expect(editor.getCurrentPageShapes().length).toBe(2)

		editor.selectAll()
		expect(editor.getSelectedShapes().length).toBe(2)

		editor.keyUp('Enter')
		editor.expectToBeIn('select.idle')
	})
})

describe('When in the pointing state', () => {
	it('Switches back to idle on cancel', () => {
		editor.setCurrentTool('geo')
		editor.pointerDown(50, 50)
		editor.expectToBeIn('geo.pointing')
		editor.cancel()
		editor.expectToBeIn('geo.idle')
	})

	it('Enters the select.resizing state on drag start', () => {
		editor.setCurrentTool('geo')
		editor.pointerDown(50, 50)
		editor.pointerMove(51, 51) // not far enough!
		editor.expectToBeIn('geo.pointing')
		editor.pointerMove(55, 55)
		editor.expectToBeIn('select.resizing')
	})

	it('Enters the select.resizing state on pointer move', () => {
		editor.setCurrentTool('geo')
		editor.pointerDown(50, 50)
		editor.cancel()
		editor.expectToBeIn('geo.idle')
	})

	it('Returns to the idle state on interrupt', () => {
		editor.setCurrentTool('geo')
		editor.pointerDown(50, 50)
		editor.interrupt()
		editor.expectToBeIn('geo.idle')
	})

	it('Creates a geo and returns to select tool on pointer up', () => {
		expect(editor.getCurrentPageShapes().length).toBe(0)
		editor.setCurrentTool('geo')
		editor.pointerDown(50, 50)
		editor.pointerUp(50, 50)
		editor.expectToBeIn('select.idle')
		expect(editor.getCurrentPageShapes().length).toBe(1)
	})

	it('Creates a geo and returns to geo.idle on pointer up if tool lock is enabled', () => {
		editor.updateInstanceState({ isToolLocked: true })
		expect(editor.getCurrentPageShapes().length).toBe(0)
		editor.setCurrentTool('geo')
		editor.pointerDown(50, 50)
		editor.pointerUp(50, 50)
		editor.expectToBeIn('geo.idle')
		expect(editor.getCurrentPageShapes().length).toBe(1)
	})
})

describe('When in the resizing state while creating a geo shape', () => {
	it('Returns to geo on cancel', () => {
		editor.setCurrentTool('geo')
		editor.pointerDown(50, 50)
		editor.pointerMove(55, 55)
		editor.expectToBeIn('select.resizing')
		editor.cancel()
		editor.expectToBeIn('geo.idle')
	})

	it('Returns to select.idle on complete', () => {
		editor.setCurrentTool('geo')
		editor.pointerDown(50, 50)
		editor.pointerMove(100, 100)
		editor.pointerUp(100, 100)
		editor.expectToBeIn('select.idle')
	})

	it('Returns to geo.idle on complete if tool lock is enabled', () => {
		editor.updateInstanceState({ isToolLocked: true })
		editor.setCurrentTool('geo')
		editor.pointerDown(50, 50)
		editor.pointerMove(100, 100)
		editor.pointerUp(100, 100)
		editor.expectToBeIn('geo.idle')
	})
})
