目录

第52章react-dnd实战拖拽版TodoList

目录

学了很多技术之后,这节来综合练习下,做个 Todo List。

当然,不是普通的那种,而是拖拽版:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/3cd18397850340d885d6fe439e900786~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1236&h=900&s=45348&e=png&b=ffffff

可以拖拽右边的 Todo Item 到列表里:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/f7c35c5b3ec14f80995cc53f9331cd3a~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1232&h=888&s=49605&e=png&b=ffffff

拖拽到空白区域的时候,会高亮标出,松手后插入到该位置。

或者也可以拖动列表中的 TodoItem 调整顺序。

还可以拖到垃圾箱删除:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/1f26eb71028f4341ba644a9ab6d31d72~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1292&h=884&s=58619&e=png&b=ffffff

当拖动过来或者双击 TodoItem 的时候,可以进入编辑模式:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/108b9733896d4134bfa592b0afd8d95d~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1198&h=866&s=47240&e=png&b=ffffff

此外,Todo Item 勾选后代表完成:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/aa4631b2f7af446390dcb1f6c98668f4~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1204&h=848&s=43906&e=png&b=fcfcfc

技术栈用 react-dnd + zustand + tailwind + react-spring。

列表的数据都在 Store 里存储:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/56b52c9131a2425b9c36a1d437f29c5d~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1342&h=1128&s=67206&e=png&b=ffffff

增删改之后修改 Store 里的数据。

用 React Dnd 来做拖拽。

用 react-spring 实现过渡动画。

样式使用 Tailwind 的原子化样式来写。

需求理清了,我们正式上手写:

npx create-vite

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/774e191a48d2488bb0d27921c1e0abaa~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=812&h=430&s=50349&e=png&b=010101

进入项目,去掉 StrictMode:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/7cbbb9a0203848e69b2f521a5201b701~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=852&h=390&s=73061&e=png&b=1f1f1f

然后新建 TodoList/index.tsx 组件:

import { FC } from "react";

interface TodoListProps {

}

export const TodoList:FC<TodoListProps> = (props) => {
    
    return <div></div>
}

按照 tailwind 文档里的步骤安装 tailwind:

npm install -D tailwindcss postcss autoprefixer

npx tailwindcss init -p

会生成 tailwind 和 postcss 配置文件:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/67806a45c69b4b3794823611ad606ee5~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=796&h=168&s=31606&e=png&b=181818

修改下 content 配置,也就是从哪里提取 className:

/** @type {import('tailwindcss').Config} */
export default {
  content: [
    "./index.html",
    "./src/**/*.{js,ts,jsx,tsx}",
  ],
  theme: {
    extend: {},
  },
  plugins: [],
}

前面 tailwind 那节讲过,tailwind 会提取 className 之后按需生成最终的 css。

改下 index.css 引入 tailwind 基础样式:

@tailwind base;
@tailwind components;
@tailwind utilities;

安装 tailwind 插件之后:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/e7dfdd931a6946cfa629042c07b10e04~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1902&h=838&s=411909&e=png&b=1d1d1d

在写代码的时候就会提示 className 和对应的样式值:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/33e4d11884844ae98b9bf27a1f6504a4~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1560&h=468&s=60658&e=png&b=202020

这个插件触发提示需要先敲一个空格,这点要注意下:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/0acf863e44ce47b189ba37ee9e7b044a~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1364&h=494&s=132209&e=gif&f=38&b=1e1e1e

有的你不知道 className 叫啥的样式,还可以在 tailwind 文档里搜:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/f7efbcb935b94005b51dbd4f4faed5b7~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1716&h=1218&s=333844&e=png&b=ffffff

改下 TodoList 的样式:

import { FC } from "react";

interface TodoListProps {

}

export const TodoList:FC<TodoListProps> = (props) => {

    return <div className="w-1000 h-600 m-auto mt-100 p-10 border-2 border-black"></div>
}

设置 width 1000,height 600,margin-top 100 padding 10 然后 border 2

在 App.tsx 引入下:

import { TodoList } from './TodoList'

function App() {

  return <TodoList></TodoList>
}

export default App

把开发服务跑起来:

npm install

npm run dev

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/4d96855574ec442ea2462f5c0631e88b~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=824&h=254&s=36452&e=png&b=191919

为啥部分样式没生效呢?

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/89fef6ecd2d94b6b8aebeffa96fe7914~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1872&h=454&s=45085&e=png&b=ffffff

因为像 w-1000 h-600 mt-100 这种,在内置的 className 里并没有。

需要在 tailwind.config.js 里配置下:

/** @type {import('tailwindcss').Config} */
export default {
  content: [
    "./index.html",
    "./src/**/*.{js,ts,jsx,tsx}",
  ],
  theme: {
    extend: {
      width: {
        1000: '1000px',
      },
      height: {
        600: '600px'
      },
      margin: {
        100: '100px'
      }
    },
  },
  plugins: [],
}

这样就好了:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/454e39769a434860b65e1031456a5ce0~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2576&h=1460&s=94974&e=png&b=ffffff

然后继续写布局:

import { FC } from "react";

interface TodoListProps {

}

export const TodoList:FC<TodoListProps> = (props) => {


    return <div className={`
            w-1000 h-600 m-auto mt-100 p-10
            border-2 border-black
            flex justify-between items-start
        `}>
        <div className="flex-2 h-full mr-10 bg-blue-400 overflow-auto">
        </div>

        <div className="flex-1 h-full bg-blue-400">

        </div>
    </div>
}

父元素 display:flex,然后 子元素分别 2 和 1 的比例,设置 margin-right:10px

这里 h-full 是 height:100%

flex-2 要配置下:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/5d31c329f41f4dc58fcbffeb9b40c419~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=940&h=966&s=115834&e=png&b=1f1f1f

看一下: https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/64307181a9474eb0a5b487c6bfc7004b~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2252&h=1408&s=78316&e=png&b=ffffff

你会发现 margin 和 padding 都不是 10px,而是 2.5rem

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/f96cc35d887548fb852dd35351e0795f~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1926&h=1082&s=151072&e=png&b=f8f3f0

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/ee6228a1307b4016994a5411cf3150ef~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1986&h=1070&s=202612&e=png&b=fefefe

我们在 tailwind.config.js 里覆盖下:

/** @type {import('tailwindcss').Config} */
export default {
  content: [
    "./index.html",
    "./src/**/*.{js,ts,jsx,tsx}",
  ],
  theme: {
    extend: {
      width: {
        1000: '1000px',
        600: '600px'
      },
      height: {
        600: '600px'
      },
      margin: {
        100: '100px',
        10: '10px'
      },
      padding: {
        10: '10px'
      },
      flex: {
        2: 2
      }
    },
  },
  plugins: [],
}

这样就好了:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/f8ef443dcd0944fe873bdb9096d59c2e~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2358&h=1386&s=82901&e=png&b=70a5f3

然后去掉背景颜色,添加 List、GarbageBin、NewItem 这三个组件:

import { FC } from "react";
import classNames from "classnames";
import { NewItem } from "./NewItem";
import { GarbageBin } from "./GarbageBin";
import { List } from "./List";

interface TodoListProps {

}

export const TodoList:FC<TodoListProps> = (props) => {


    return <div className={classNames(
            "w-1000 h-600 m-auto mt-100 p-10",
            "border-2 border-black",
            "flex justify-between items-start"
        )}>
        <div className="flex-2 h-full mr-10 overflow-auto">
            <List/>
        </div>

        <div className={classNames(
            "flex-1 h-full",
            "flex flex-col justify-start"
        )}>
            <NewItem/>
            <GarbageBin className={"mt-100"}/>
        </div>
    </div>
}

这里多行 className 换成用 classnames 包来写。

npm install --save classnames

分别添加 GarbageBin.tsx

import classNames from "classnames"
import { FC } from "react"

interface GarbaseProps{
    className?: string | string[]
}

export const GarbageBin: FC<GarbaseProps> = (props) => {
    
    const cs = classNames(
        "h-100 border-2 border-black",
        props.className
    );

    return <div className={cs}></div>
}

NewItem.tsx

import classNames from "classnames"
import { FC } from "react"

interface NewItemProps{
    className?: string | string[]
}

export const NewItem: FC<NewItemProps> = (props) => {
    
    const cs = classNames(
        "h-200 border-2 border-black",
        props.className
    );

    return <div className={cs}></div>
}

还有 List.tsx

import classNames from "classnames"
import { FC } from "react"

interface ListProps{
    className?: string | string[]
}

export const List: FC<ListProps> = (props) => {
    
    const cs = classNames(
        "h-full border-2 border-black",
        props.className
    );

    return <div className={cs}></div>
}

这里的 h-200、h-100 要在配置文件里加一下:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/257e418492a442be90be1932c4f70252~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=942&h=840&s=113060&e=png&b=1f1f1f

现在界面是这样的:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/3492fc27e4174ac998c55513d8b0037e~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2146&h=1384&s=68611&e=png&b=ffffff

然后先来实现 List 组件部分:

import classNames from "classnames"
import { FC } from "react"

interface ListProps{
    className?: string | string[]
}

export const List: FC<ListProps> = (props) => {
    
    const cs = classNames(
        "h-full p-10",
        props.className
    );

    return <div className={cs}>
        <Item/>
        <Item/>
        <Item/>
        <Item/>
        <Item/>
        <Item/>
        <Item/>
    </div>
}

function Item() {
    return <div className={classNames(
        "h-100 border-2 border-black bg-blue-300 mb-10 p-10",
        "flex justify-start items-center",
        "text-xl tracking-wide"
    )}>
        <input type="checkbox" className="w-40 h-40 mr-10"/>
        <p>待办事项</p>
    </div>
}

配置文件加一下 w-40、h-40 的配置:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/f90e80e192ea4a2284b6b41303958b61~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=390&h=476&s=45148&e=png&b=202020

看下效果:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/75220b2add3d4dbb9b238a1d94e6517c~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2002&h=1340&s=606256&e=gif&f=23&b=fefefe

里面用到的 className 可以去查 tailwind 文档

然后是 NewItem 组件:

import classNames from "classnames"
import { FC } from "react"

interface NewItemProps{
    className?: string | string[]
}

export const NewItem: FC<NewItemProps> = (props) => {
    
    const cs = classNames(
        "h-100 border-2 border-black",
        "leading-100 text-center text-2xl",
        "bg-green-300",
        "cursor-move select-none",
        props.className
    );

    return <div className={cs}>新的待办事项</div>
}

GarbageBin 组件:

import classNames from "classnames"
import { FC } from "react"

interface GarbaseProps{
    className?: string | string[]
}

export const GarbageBin: FC<GarbaseProps> = (props) => {
    
    const cs = classNames(
        "h-200 border-2 border-black",
        "bg-orange-300",
        "leading-200 text-center text-2xl",
        "cursor-move select-none",
        props.className
    );

    return <div className={cs}>垃圾箱</div>
}

在配置文件里加一下两个 line-height:

/** @type {import('tailwindcss').Config} */
export default {
  content: [
    "./index.html",
    "./src/**/*.{js,ts,jsx,tsx}",
  ],
  theme: {
    extend: {
      width: {
        1000: '1000px',
        600: '600px',
        40: '40px'
      },
      height: {
        600: '600px',
        200: '200px',
        100: '100px',
        40: '40px'
      },
      margin: {
        100: '100px',
        10: '10px'
      },
      padding: {
        10: '10px'
      },
      flex: {
        2: 2
      },
      lineHeight: {
        100: '100px',
        200: '200px'
      }
    },
  },
  plugins: [],
}

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/0b63c9785532493985958958c7ebfc9f~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2040&h=1292&s=94882&e=png&b=ffffff

其实这些 width、height、margin、padding 的值的覆盖可以统一放到 spacing 里:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/31d7bfb040974c39bbd86e27965a15a1~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=828&h=1316&s=136553&e=png&b=1f1f1f

/** @type {import('tailwindcss').Config} */
export default {
  content: [
    "./index.html",
    "./src/**/*.{js,ts,jsx,tsx}",
  ],
  theme: {
    extend: {
      spacing: {
        10: '10px',
        40: '40px',
        100: '100px',
        200: '200px',
        600: '600px',
        1000: '1000px',
      },
      width: {
        // 1000: '1000px',
        // 600: '600px',
        // 40: '40px',
        // 10: '10px'
      },
      height: {
        // 600: '600px',
        // 200: '200px',
        // 100: '100px',
        // 40: '40px',
        // 10: '10px'
      },
      margin: {
        // 100: '100px',
        // 10: '10px'
      },
      padding: {
        // 10: '10px'
      },
      flex: {
        2: 2
      },
      lineHeight: {
        100: '100px',
        200: '200px'
      }
    },
  },
  plugins: [],
}

tailwind 文档里写了,很多样式都继承 spacing 的配置:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/4f246870b60b49658c303ab2119e6fde~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1476&h=274&s=63515&e=png&b=ffffff

或者不想全局改默认配置,也可以用 text-[14px] 这种方式。

text-[14px] 就会生成 font-size:14px 的样式:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/82af77d4f8da48dc8a4a742a762e4398~tplv-k3u1fbpfcp-watermark.image?

接下来加上 react-dnd 来做拖拽。

安装用到的包:

npm install react-dnd react-dnd-html5-backend

在 main.tsx 引入下 DndProvider

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/6c015d32821849f1825cb5e52a76bcd8~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1036&h=448&s=97879&e=png&b=1f1f1f

import ReactDOM from 'react-dom/client'
import App from './App.tsx'
import './index.css'
import { DndProvider } from 'react-dnd'
import { HTML5Backend } from 'react-dnd-html5-backend'

ReactDOM.createRoot(document.getElementById('root')!).render(
    <DndProvider backend={HTML5Backend}>
      <App />
    </DndProvider>
)

它是 react-dnd 用来跨组件传递数据的。

在 NewItem.tsx 组件里用 useDrag 添加拖拽:

import classNames from "classnames"
import { FC, useEffect, useRef } from "react"
import { useDrag } from "react-dnd";

interface NewItemProps{
    className?: string | string[]
}

export const NewItem: FC<NewItemProps> = (props) => {

    const ref = useRef<HTMLDivElement>(null);

    const [{ dragging }, drag] = useDrag({
        type: 'new-item',
        item: {},
        collect(monitor) {
            return {
                dragging: monitor.isDragging()
            }
        }
    });

    useEffect(() => {
        drag(ref);
    }, []);

    const cs = classNames(
        "h-100 border-2 border-black",
        "leading-100 text-center text-2xl",
        "bg-green-300",
        "cursor-move select-none",
        dragging ? 'border-dashed bg-white' : '',
        props.className
    );

    return <div ref={ref} className={cs}>新的待办事项</div>
}

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/ef1e472b1c294f4a829b2ed8d4b6d23f~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=998&h=1046&s=172790&e=png&b=1f1f1f

拖动过程中,设置 border 虚线、背景白色。

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/9122542d162040d4a0354f5c04dac837~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2002&h=1340&s=579987&e=gif&f=27&b=fefefe

然后在 List 的 Item 也加上 useDrag 拖拽:

function Item() {
    const ref = useRef<HTMLDivElement>(null);

    const [{ dragging }, drag] = useDrag({
        type: 'list-item',
        item: {},
        collect(monitor) {
            return {
                dragging: monitor.isDragging()
            }
        }
    });

    useEffect(() => {
        drag(ref);
    }, []);

    return <div ref={ref} className={classNames(
        "h-100 border-2 border-black bg-blue-300 mb-10 p-10",
        "flex justify-start items-center",
        "text-xl tracking-wide",
        dragging ? 'bg-white border-dashed' : ''
    )}>
        <input type="checkbox" className="w-10 h-10 mr-10"/>
        <p>待办事项</p>
    </div>
}

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/26da96e8906f4139834a84d8ac6b0566~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2002&h=1340&s=1233221&e=gif&f=39&b=fefefe

在垃圾箱添加 useDrop:

import classNames from "classnames"
import { FC, useEffect, useRef } from "react"
import { useDrop } from "react-dnd";

interface GarbaseProps{
    className?: string | string[]
}

export const GarbageBin: FC<GarbaseProps> = (props) => {
    
    const ref = useRef<HTMLDivElement>(null);

    const [{ isOver }, drop] = useDrop(() => {
        return {
            accept: 'list-item',
            drop(item) {},
            collect(monitor) {
                return {
                    isOver: monitor.isOver()
                }
            }
        }
    });

    useEffect(()=> {
        drop(ref);
    }, []);
    

    const cs = classNames(
        "h-200 border-2 border-black",
        "bg-orange-300",
        "leading-200 text-center text-2xl",
        "cursor-move select-none",
        isOver ? "bg-yellow-400 border-dashed" : "",
        props.className
    );

    return <div ref={ref} className={cs}>垃圾箱</div>
}

accept 指定了 list-item,只有对应的 type 拖拽到这里才能触发 isOver:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/309f71b6ffaf49e6bd6cff0c49e5a791~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2002&h=1340&s=768479&e=gif&f=42&b=fefefe

那新的 todo item 拖到哪里呢?

到这里: https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/5510f73bb0454c99aaa847abce813b40~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1268&h=1170&s=64789&e=png&b=9dc4f8

所以我们要把这些地方也新建个组件,然后添加 useDrop:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/e41dd6fb69fc4812a660cf4f4cc5c952~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=886&h=1156&s=149891&e=png&b=1f1f1f

去掉之前 Item 的 mt-10 换成 Gap 的 h-10:

import classNames from "classnames"
import { FC, useEffect, useRef } from "react"
import { useDrag, useDrop } from "react-dnd";

interface ListProps{
    className?: string | string[]
}

export const List: FC<ListProps> = (props) => {
    
    const cs = classNames(
        "h-full p-10",
        props.className
    );

    return <div className={cs}>
        <Gap/>
        <Item/>
        <Gap/>
        <Item/>
        <Gap/>
        <Item/>
        <Gap/>
        <Item/>
        <Gap/>
        <Item/>
        <Gap/>
        <Item/>
        <Gap/>
        <Item/>
        <Gap/>
    </div>
}

function Gap() {
    const ref = useRef<HTMLDivElement>(null);

    const [{ isOver }, drop] = useDrop(() => {
        return {
            accept: 'new-item',
            drop(item) {},
            collect(monitor) {
                return {
                    isOver: monitor.isOver()
                }
            }
        }
    });

    useEffect(()=> {
        drop(ref);
    }, []);

    const cs = classNames(
        "h-10",
        isOver ? 'bg-red-300' : ''
    );

    return <div ref={ref} className={cs}></div>
}

function Item() {
    const ref = useRef<HTMLDivElement>(null);

    const [{ dragging }, drag] = useDrag({
        type: 'list-item',
        item: {},
        collect(monitor) {
            return {
                dragging: monitor.isDragging()
            }
        }
    });

    useEffect(() => {
        drag(ref);
    }, []);

    return <div ref={ref} className={classNames(
        "h-100 border-2 border-black bg-blue-300 p-10",
        "flex justify-start items-center",
        "text-xl tracking-wide",
        dragging ? 'bg-white border-dashed' : ''
    )}>
        <input type="checkbox" className="w-40 h-40 mr-10"/>
        <p>待办事项</p>
    </div>
}

覆盖下 w-10、h-10 的值,默认是 rem,我们还是用 px:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/18f65b3c4b9642bfae9671b381602efa~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=510&h=544&s=53837&e=png&b=1f1f1f

现在 new-item 就能拖过来了:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/df32c78d870c4a0fa59ed67af8e40aef~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2002&h=1340&s=918893&e=gif&f=42&b=fefefe

现在 Gap 和 Item 代码挺多了,分离出去作为单独的模块 Gap.tsx 和 Item.tsx

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/89b466bf51cc4bb58fa8d787a796e8d6~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=574&h=314&s=36415&e=png&b=191919

接下来处理下具体的状态逻辑。

安装 zustand:

npm install --save zustand

创建 TodoList/Store.ts

import { create } from 'zustand';

export interface ListItem {
  id: string,
  status: 'todo' | 'done',
  content: string
}

type State = {
  list: Array<ListItem>
}

type Action = {
  addItem: (item: ListItem) => void,
  deleteItem: (id: string) => void,
  updateItem: (item: ListItem) => void,
}

export const useTodoListStore = create<State & Action>((set) => ({
  list: [],
  addItem: (item: ListItem) => {
    set((state) => {
      return {
        list: [
          ...state.list,
          item
        ]
      }
    })
  },
  deleteItem: (id: string) => {
    set((state) => {
      return {
        list: state.list.filter(item => {
          return item.id !== id;
        })
      }
    });
  },
  updateItem: (updateItem: ListItem) => {
    set(state => {
      return {
        list: state.list.map(item => {
          if(item.id === updateItem.id) {
            return updateItem;
          }
          return item;
        })
      }
    })
  }
}))

state 就是 list,然后添加 addItem、deleteItem、updateItem 的方法。

在 List 组件里引入下:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/200db6d122df449ea87562e66cf76423~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1050&h=1148&s=192945&e=png&b=1f1f1f

传入 data,顺便指定 key:

import classNames from "classnames"
import { FC, Fragment} from "react"
import { Gap } from "./Gap";
import { Item } from "./Item";
import { useTodoListStore } from "./store";

interface ListProps{
    className?: string | string[]
}

export const List: FC<ListProps> = (props) => {
    
    const list = useTodoListStore(state => state.list);

    const cs = classNames(
        "h-full p-10",
        props.className
    );

    return <div className={cs}>
        {
            list.length ? list.map(item => {
                return <Fragment key={item.id}>
                    <Gap/>
                    <Item data={item} />
                </Fragment>
            }) : '暂无待办事项'
        }
        <Gap/>
    </div>
}

<Fragment> 也可以写 <></>,它只是用来给多个 children 包一层,但不会生成 dom 节点。

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/49a36142c31047c3baaf44f32659ca7f~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1884&h=618&s=96995&e=png&b=fefefe

在 Item 组件添加 content 参数:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/c4ed62ad61634057944b111b94b7df8e~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1030&h=1308&s=207133&e=png&b=1f1f1f

看下效果:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/892b6019331f4262adcf0c5613694708~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1990&h=1274&s=71741&e=png&b=ffffff

我们加一下添加 item 的处理:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/f2e5a7d6a960428295dd01c7044c553c~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1160&h=1456&s=230823&e=png&b=1f1f1f

import classNames from "classnames";
import { useEffect, useRef } from "react";
import { useDrop } from "react-dnd";
import { useTodoListStore } from "./store";

export function Gap() {
    const addItem = useTodoListStore(state => state.addItem);

    const ref = useRef<HTMLDivElement>(null);

    const [{ isOver }, drop] = useDrop(() => {
        return {
            accept: 'new-item',
            drop(item) {
                addItem({
                    id: Math.random().toString().slice(2, 8),
                    status: 'todo',
                    content: '待办事项'
                });
            },
            collect(monitor) {
                return {
                    isOver: monitor.isOver()
                }
            }
        }
    });

    useEffect(()=> {
        drop(ref);
    }, []);

    const cs = classNames(
        "h-10",
        isOver ? 'bg-red-300' : ''
    );

    return <div ref={ref} className={cs}></div>
}

这里用 Math.random 生成 6 位的随机数:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/e879dae6fbee4f068cb30617ce49baa0~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=574&h=246&s=33927&e=png&b=010101

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/e1ca19e53c0444cdbe8c1a3690ed1282~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2002&h=1340&s=997938&e=gif&f=59&b=fefefe

然后加一下删除的处理:

drag 的时候加上传递的数据:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/bc2f71b87d03493e926c314c631b43c6~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=948&h=996&s=140816&e=png&b=1f1f1f

drop 的时候拿到 id 执行删除:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/32f22663674a4695866e50becb226347~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1090&h=1034&s=187657&e=png&b=1f1f1f

测试下:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/49a887cbabc34f90aa9219eb52591bb8~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2002&h=1186&s=748868&e=gif&f=41&b=fefefe

删除也没问题。

然后加上编辑功能:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/ff51214d868e41fd803d5c92702aac86~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1332&h=1460&s=274515&e=png&b=1f1f1f

用两个 state 分别保存 editing 状态和 input 内容。

onDoubleClick 的时候显示 input,修改 editing 状态为 true。

onBlur 的时候修改 editing 状态为 false。

并且用 updateItem 更新状态:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/ecdc7a6bf98046578b925362f6b1f826~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1052&h=1328&s=218488&e=png&b=1f1f1f

没啥问题:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/9c9e72197bba430db45aa699c20579e3~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2002&h=1186&s=98291&e=gif&f=30&b=fefefe

然后当选中 checkbox 的时候,也要 updateItem:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/e8bbcfa0b97b4804b71f71ec6f398351~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1082&h=906&s=170370&e=png&b=1f1f1f

import classNames from "classnames";
import { useEffect, useRef, useState } from "react";
import { useDrag } from "react-dnd";
import { ListItem, useTodoListStore } from "./store";

interface ItemProps {
    data: ListItem
}

export function Item(props: ItemProps) {

    const {
        data
    } = props;

    const updateItem = useTodoListStore(state => state.updateItem);

    const ref = useRef<HTMLDivElement>(null);

    const [editing, setEditing] = useState(false);

    const [editingContent, setEditingContent] = useState(data.content);

    const [{ dragging }, drag] = useDrag({
        type: 'list-item',
        item: {
            id: data.id
        },
        collect(monitor) {
            return {
                dragging: monitor.isDragging()
            }
        }
    });

    useEffect(() => {
        drag(ref);
    }, []);

    return <div ref={ref} className={classNames(
            "h-100 border-2 border-black bg-blue-300 p-10",
            "flex justify-start items-center",
            "text-xl tracking-wide",
            dragging ? 'bg-white border-dashed' : ''
        )}
        onDoubleClick={() => {
            setEditing(true)
        }}
    >
        <input 
            type="checkbox" 
            className="w-40 h-40 mr-10"
            checked={data.status === 'done' ? true : false}
            onChange={(e) => {
                updateItem({
                    ...data,
                    status: e.target.checked ? 'done' : 'todo'
                })
            }}
        />
        <p>
            {
                editing ? <input 
                    value={editingContent}
                    onChange={(e) => {
                        setEditingContent(e.target.value)
                    }}
                    onBlur={() => {
                        setEditing(false);
                        updateItem({
                            ...data,
                            content: editingContent
                        })
                    }}
                /> : data.content 
            }
        </p>
    </div>
}

还有,现在不管拖动到哪里都是在后面插入:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/b960e0c5576a44cbbf3924c1c6536d97~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2002&h=1186&s=275684&e=gif&f=26&b=fefefe

我们希望能根据 drop 的位置来插入:

所以给 Gap 传入 id 参数:

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/29d52753c42e411d9348462282adc234~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=854&h=746&s=104035&e=png&b=1f1f1f

然后 Gap 组件 drop 的时候传入 addItem 方法:

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/b5a44ccbee6842e29c01265d5767af0a~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1056&h=1074&s=182804&e=png&b=1f1f1f

addItem 方法里根据 id 插入:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/44f75b919d664cc5a382d8449ae05082~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1080&h=1166&s=144674&e=png&b=1f1f1f

没有传就插入在后面,否则 findIndex,然后在那个位置插入。

测试下:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/503e5f15efd743b58a4cc9b48dea2219~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2002&h=1186&s=807728&e=gif&f=51&b=fefefe

没啥问题。

不过 gap 区域有点小,大家实现的时候可以改大一点。

还有,现在一刷新,数据就没了:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/4830405c89b4432da079e6e877751b5f~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2220&h=1476&s=171020&e=gif&f=32&b=fefefe

我们给 zustand 加上 persist 中间件:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/549cda672ac040109670a737a84678b8~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1322&h=820&s=144061&e=png&b=1f1f1f

注意,ts + middleware 的场景,zustand 要换这种写法。

文档的解释是为了更好的处理类型:

https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/9f5b5414e3514e7b93f1586669e30b6b~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2042&h=764&s=187497&e=png&b=f6f8fa

反正功能是一样的。

import { StateCreator, create } from 'zustand';
import { persist } from 'zustand/middleware';

export interface ListItem {
  id: string,
  status: 'todo' | 'done',
  content: string
}

type State = {
  list: Array<ListItem>
}

type Action = {
  addItem: (item: ListItem, id?: string) => void,
  deleteItem: (id: string) => void,
  updateItem: (item: ListItem) => void,
}

const stateCreator: StateCreator<State & Action> = (set) => ({
  list: [],
  addItem: (item: ListItem, id?: string) => {
    set((state) => {
      if(!id) {
        return {
          list: [
            ...state.list,
            item
          ]
        }
      }

      const newList = [
        ...state.list, 
      ];

      const index = newList.findIndex(item => item.id === id);

      newList.splice(index, 0, item);

      return {
        list: newList
      }
    })
  },
  deleteItem: (id: string) => {
    set((state) => {
      return {
        list: state.list.filter(item => {
          return item.id !== id;
        })
      }
    });
  },
  updateItem: (updateItem: ListItem) => {
    set(state => {
      return {
        list: state.list.map(item => {
          if(item.id === updateItem.id) {
            return updateItem;
          }
          return item;
        })
      }
    })
  }
});

export const useTodoListStore = create<State & Action>()(persist(stateCreator, {
  name: 'todolist'
}));

测试下:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/ff437e925fa345dea82abc2d92f1e664~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2524&h=1314&s=245754&e=png&b=fdfdfd

https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/77796c9bfb614e48a2375a8cb3e1b3e8~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2220&h=1476&s=484880&e=gif&f=49&b=fcfcfc

现在,数据就被保存到了 localstorage 中,刷新数据也不会丢失。

这样,拖拽版 TodoList 就完成了。

大家还可以加个拖拽排序功能,和上节实现一样。

最后,我们加上过渡动画,用 react-spring:

npm install --save @react-spring/web

然后渲染 list 的时候用 react-spring 的 useTransition 的 hook 处理下:

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/e982999a8d9a469cb8de6f84a2d6e06d~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1060&h=1032&s=176486&e=png&b=202020

useTransition 会根据传入的配置来生成 style,这些 style 要加在 animated.div 上。

并且,keys 也是在配置里传入的,animated.div 会自动添加。

https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/56ee989897404563b44fe1557dc99f2c~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=2220&h=1476&s=892070&e=gif&f=61&b=fefefe

案例代码上传了小册仓库

总结

我们用 react-dnd + zustand 实现了拖拽版 todolist。

用 tailwind 来写的样式。

用 @react-spring/web 加上了过渡动画。

这是个综合实战,对 react-dnd、tailwind、zustand、react-spring 都有较全面的应用。