详解Jotai Immer如何实现undo redo功能示例详解

  

详解Jotai Immer如何实现undo redo功能示例详解

Jotai Immer是一个结合了Jotai和Immer两种状态管理库的工具,其中Immer提供了基于不可变数据结构的状态修改方式,Jotai则提供了状态的管理和更新功能。通过结合使用两个库,我们可以更加方便地进行状态管理,并实现undo redo功能。

安装和引入

首先,我们需要安装Jotai Immer库:

npm install jotai-immer

接着,在需要使用的文件中引入:

import { Provider, useAtom } from 'jotai'
import { atomWithImmer } from 'jotai-immer'

创建包含undo redo功能的Atom

通过atomWithImmer函数创建一个带有undo redo功能的Atom:

function undoRedoAtom(defaultValue) {
  const history = [defaultValue]
  let pointer = 0

  return atomWithImmer(defaultValue, (get, set, op) => {
    if (op.type === 'UNDO') {
      pointer--
      if (pointer < 0) {
        pointer = 0
      }
      set(history[pointer])
    } else if (op.type === 'REDO') {
      pointer++
      if (pointer >= history.length) {
        pointer = history.length - 1
      }
      set(history[pointer])
    } else {
      op = typeof op === 'function' ? op : op?.newState !== undefined ? () => op.newState : () => op
      const nextState = produce(get(), op)
      if (nextState !== history[pointer]) {
        pointer++
        history.splice(pointer, history.length - pointer, nextState)
      }
      set(nextState)
    }
  })
}

这个函数的实现主要使用了数组来保存历史状态,并依靠produce函数来实现基于不可变数据结构的状态修改。

使用Atom创建State

使用undoRedoAtom函数创建带有undo redo功能的State,并在组件中使用:

const myAtom = undoRedoAtom({ count: 0 })
function MyComponent() {
  const [state, setState] = useAtom(myAtom)

  const incrementCount = () => setState((draft) => {
    draft.count++
  })

  const decrementCount = () => setState((draft) => {
    draft.count--
  })

  const undo = () => setState({ type: 'UNDO' })

  const redo = () => setState({ type: 'REDO' })

  return (
    <div>
      <div>Count: {state.count}</div>
      <button onClick={incrementCount}>+</button>
      <button onClick={decrementCount}>-</button>
      <button onClick={undo} disabled={historyPointer === 0}>Undo</button>
      <button onClick={redo} disabled={historyPointer === history.length-1}>Redo</button>
    </div>
  )
}

在这个示例中,我们使用useAtom钩子获取状态和setState函数,并在组件中实现了一个计数器和undo redo按钮的功能。

示例二:在表单中实现undo redo功能

我们可以在包含表单的组件中使用atomWithImmer来实现undo redo功能:

const formAtom = atomWithImmer({
  name: '',
  age: '',
  address: '',
})

function Form() {
  const [form, setForm] = useAtom(formAtom)

  const undo = () => setForm({ type: 'UNDO' })
  const redo = () => setForm({ type: 'REDO' })

  const handleChange = (e) => {
    const { name, value } = e.target
    setForm((draft) => {
      draft[name] = value
    })
  }

  return (
    <form>
      <div>
        <label htmlFor="name">Name:</label>
        <input type="text" name="name" value={form.name} onChange={handleChange} />
      </div>
      <div>
        <label htmlFor="age">Age:</label>
        <input type="number" name="age" value={form.age} onChange={handleChange} />
      </div>
      <div>
        <label htmlFor="address">Address:</label>
        <input type="text" name="address" value={form.address} onChange={handleChange} />
      </div>
      <button onClick={undo} disabled={historyPointer === 0}>Undo</button>
      <button onClick={redo} disabled={historyPointer === history.length-1}>Redo</button>
    </form>
  )
}

在这个示例中,我们实现了一个包含表单的组件,并使用useAtom钩子获取状态和setState函数。通过实现handleChange函数,来实现表单内容的更新。同时,通过实现undoredo函数,来实现undo redo功能的实现。

总结:

通过使用Jotai Immer提供的atomWithImmer函数,我们可以方便地实现undo redo功能。无论是计数器还是表单,只要使用了State,都可以通过这种方式来实现撤销恢复相关的操作。

相关文章