import {useReducer} from 'react'; import {reactHooks} from 'sentry-test/reactTestingLibrary'; import {makeCombinedReducers} from 'sentry/utils/useCombinedReducer'; import { makeUndoableReducer, UndoableNode, useUndoableReducer, } from 'sentry/utils/useUndoableReducer'; describe('makeUndoableReducer', () => { it('does not overflow undo/redo', () => { const mockFirstReducer = jest.fn().mockImplementation(v => ++v); const reducer = makeUndoableReducer(mockFirstReducer); expect(() => reducer({previous: undefined, current: 0, next: undefined}, {type: 'undo'}) ).not.toThrow(); expect(() => reducer({previous: undefined, current: 0, next: undefined}, {type: 'redo'}) ).not.toThrow(); }); it('calls undo/redo if action matches', () => { const mockFirstReducer = jest.fn().mockImplementation(v => { return ++v; }); const reducer = makeUndoableReducer(mockFirstReducer); const first: UndoableNode = { previous: undefined, current: 0, next: undefined, }; const current = { previous: first, current: 1, next: undefined, }; first.next = current; expect(reducer(first, {type: 'redo'})).toEqual(current); expect(reducer(current, {type: 'undo'})).toEqual(first); expect(mockFirstReducer).not.toHaveBeenCalled(); expect(reducer(current, 'add')).toEqual({ previous: current, current: 2, next: undefined, }); expect(mockFirstReducer).toHaveBeenLastCalledWith(current.current, 'add'); }); describe('useUndoableReducer', () => { it('initializes with init state', () => { const reducer = jest .fn() .mockImplementation((state: number, action: 'add' | 'subtract') => action === 'add' ? state + 1 : state - 1 ); const {result} = reactHooks.renderHook( (args: Parameters) => useUndoableReducer(args[0], args[1]), { initialProps: [reducer, 100], } ); expect(reducer).not.toHaveBeenCalled(); expect(result.current[0]).toEqual(100); }); it('updates state', () => { const reducer = jest .fn() .mockImplementation((state: number, action: 'add' | 'subtract') => action === 'add' ? state + 1 : state - 1 ); const {result} = reactHooks.renderHook( (args: Parameters) => useUndoableReducer(args[0], args[1]), {initialProps: [reducer, 0]} ); reactHooks.act(() => result.current[1]('add')); expect(result.current[0]).toEqual(1); expect(reducer).toHaveBeenNthCalledWith(1, 0, 'add'); reactHooks.act(() => result.current[1]('add')); expect(result.current[0]).toEqual(2); expect(reducer).toHaveBeenNthCalledWith(2, 0, 'add'); }); it('can undo state', () => { const {result} = reactHooks.renderHook( (args: Parameters) => useUndoableReducer(args[0], args[1]), {initialProps: [jest.fn().mockImplementation(s => s + 1), 0]} ); reactHooks.act(() => result.current[1](0)); expect(result.current[0]).toEqual(1); reactHooks.act(() => result.current[1]({type: 'undo'})); expect(result.current[0]).toEqual(0); }); it('can redo state', () => { const {result} = reactHooks.renderHook( (args: Parameters) => useUndoableReducer(args[0], args[1]), {initialProps: [jest.fn().mockImplementation(s => s + 1), 0]} ); reactHooks.act(() => result.current[1](0)); // 0 + 1 reactHooks.act(() => result.current[1](1)); // 1 + 1 reactHooks.act(() => result.current[1]({type: 'undo'})); // 2 -> 1 reactHooks.act(() => result.current[1]({type: 'undo'})); // 1 -> 0 expect(result.current[0]).toEqual(0); reactHooks.act(() => result.current[1]({type: 'redo'})); // 0 -> 1 reactHooks.act(() => result.current[1]({type: 'redo'})); // 1 -> 2 expect(result.current[0]).toEqual(2); reactHooks.act(() => result.current[1]({type: 'redo'})); // 2 -> undefined expect(result.current[0]).toEqual(2); }); }); it('can peek previous and next state', () => { const simpleReducer = (state, action) => action.type === 'add' ? state + 1 : state - 1; const {result} = reactHooks.renderHook( (args: Parameters) => useUndoableReducer(args[0], args[1]), { initialProps: [simpleReducer, 0], } ); reactHooks.act(() => result.current[1]({type: 'add'})); expect(result.current?.[2].previousState).toEqual(0); reactHooks.act(() => result.current[1]({type: 'undo'})); expect(result.current?.[2].nextState).toEqual(1); }); it('can work with primitives', () => { const simpleReducer = (state: number, action: {type: 'add'} | {type: 'subtract'}) => action.type === 'add' ? state + 1 : state - 1; const {result} = reactHooks.renderHook( (args: Parameters) => useReducer(args[0], args[1]), { initialProps: [ makeUndoableReducer(makeCombinedReducers({simple: simpleReducer})), { previous: undefined, current: { simple: 0, }, next: undefined, }, ], } ); reactHooks.act(() => result.current[1]({type: 'add'})); expect(result.current[0].current.simple).toBe(1); reactHooks.act(() => result.current[1]({type: 'undo'})); expect(result.current[0].current.simple).toBe(0); reactHooks.act(() => result.current[1]({type: 'redo'})); expect(result.current[0].current.simple).toBe(1); }); it('can work with objects', () => { const combinedReducers = makeCombinedReducers({ math: (state: number, action: {type: 'add'} | {type: 'subtract'}) => action.type === 'add' ? state + 1 : state - 1, }); const {result} = reactHooks.renderHook( (args: Parameters) => useReducer(args[0], args[1]), { initialProps: [ makeUndoableReducer(combinedReducers), { previous: undefined, current: { math: 0, }, next: undefined, }, ], } ); reactHooks.act(() => result.current[1]({type: 'add'})); expect(result.current[0].current.math).toBe(1); reactHooks.act(() => result.current[1]({type: 'undo'})); expect(result.current[0].current.math).toBe(0); reactHooks.act(() => result.current[1]({type: 'redo'})); expect(result.current[0].current.math).toBe(1); }); });