0

0

Svelte 与 SortableJS 动态多列表拖拽排序的正确实践

心靈之曲

心靈之曲

发布时间:2025-12-29 18:46:21

|

875人浏览过

|

来源于php中文网

原创

Svelte 与 SortableJS 动态多列表拖拽排序的正确实践

本文详解如何在 svelte 中结合 sortablejs 实现多个动态嵌套列表的稳定拖拽排序,重点解决因缺失 key、状态同步不一致导致的 ui 错乱、重复移动等问题,并提供基于 action 的简洁、可维护方案。

在 Svelte 中集成 SortableJS 处理多组动态列表(如分类看板)时,常见问题并非来自 SortableJS 本身,而是 Svelte 的响应式更新机制与 DOM 同步逻辑未对齐所致。最典型的症状包括:列表项“双倍移动”、拖拽后顺序回滚、跨列表拖入失败或 UI 滞后——这些几乎都指向两个核心缺陷:#each 缺失唯一 key组件状态与顶层数据源双向绑定失控

✅ 正确做法一:强制 #each 使用稳定 key

Svelte 的 #each 块默认使用数组索引作为隐式 key。当列表顺序变化(尤其跨列表拖拽)时,索引快速重排会导致 DOM 元素被错误复用或销毁,引发视觉抖动与状态错位。必须显式指定唯一、稳定、不可变的 key:

{#each category as item (item.id)}
  
  • {item.name}
  • {/each}

    (item.id) 是关键——它告诉 Svelte:“此

    KAIZAN.ai
    KAIZAN.ai

    使用AI来改善客户服体验,提高忠诚度

    下载
  • 始终绑定到 id 为该值的 item”,无论其在数组中位置如何变化,DOM 节点都会被精准复用,避免无谓的创建/销毁。

    ✅ 正确做法二:用 use: action 替代自定义组件封装

    将 Sortable 初始化逻辑封装进独立组件(如 List.svelte)看似模块化,实则引入了额外的状态层(arr = fullArr[index])、生命周期耦合(onMount 时机)及响应式陷阱(直接赋值 fullArr[index] = ... 可能绕过 Svelte 的响应式追踪)。更优解是使用 Svelte Action —— 它天然绑定到 DOM 元素,生命周期清晰,且与顶层状态完全解耦:

    
    
    {#each items as category, i}
      

    Category {i}

      {#each category as item (item.id)}
    • {item.name}
    • {/each}
    {/each}
    {JSON.stringify(items, null, 2)}
    ? 为什么 items = [...items] 不可省略?items[fromIdx].splice(...) 和 items[toIdx].splice(...) 直接修改了嵌套数组的内部结构,但 Svelte 不会自动检测深层嵌套变更。重新赋值 items = [...items] 触发顶层引用变更,强制 Svelte 重新 diff 整个 items 结构,确保 UI 精确同步。

    ✅ 进阶建议:使用 onSort 还是 onEnd?

    • onSort: 适合仅需单列表内排序,通过 sortable.toArray() 获取 ID 序列再映射回对象。但跨列表时需遍历 items.flat() 查找,性能略低且易受重复 ID 影响。
    • onEnd: 推荐用于多列表场景。event.from/event.to 明确指示来源与目标容器,event.oldIndex/event.newIndex 提供精确位置,配合 .splice() 可实现原子级、零副作用的数据迁移,逻辑更清晰、性能更优。

    ? 总结:三大避坑原则

    1. 永远为 #each 显式声明 (key) —— 尤其涉及拖拽、增删、排序等 DOM 重排操作;
    2. 优先选用 use: action 而非自定义组件封装第三方库 —— 减少状态桥接、避免生命周期冲突;
    3. 深层嵌套数组变更后,必须触发顶层响应式更新 —— arr = [...arr] 或 arr = structuredClone(arr) 是安全写法。

    遵循以上实践,即可构建出响应灵敏、行为确定、易于维护的 Svelte + SortableJS 多列表拖拽系统。

  • 相关专题

    更多
    DOM是什么意思
    DOM是什么意思

    dom的英文全称是documentobjectmodel,表示文件对象模型,是w3c组织推荐的处理可扩展置标语言的标准编程接口;dom是html文档的内存中对象表示,它提供了使用javascript与网页交互的方式。想了解更多的相关内容,可以阅读本专题下面的文章。

    2974

    2024.08.14

    li是什么元素
    li是什么元素

    li是HTML标记语言中的一个元素,用于创建列表。li代表列表项,它是ul或ol的子元素,li标签的作用是定义列表中的每个项目。本专题为大家li元素相关的各种文章、以及下载和课程。

    415

    2023.08.03

    Golang gRPC 服务开发与Protobuf实战
    Golang gRPC 服务开发与Protobuf实战

    本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

    8

    2026.01.15

    公务员递补名单公布时间 公务员递补要求
    公务员递补名单公布时间 公务员递补要求

    公务员递补名单公布时间不固定,通常在面试前,由招录单位(如国家知识产权局、海关等)发布,依据是原入围考生放弃资格,会按笔试成绩从高到低递补,递补考生需按公告要求限时确认并提交材料,及时参加面试/体检等后续环节。要求核心是按招录单位公告及时响应、提交材料(确认书、资格复审材料)并准时参加面试。

    38

    2026.01.15

    公务员调剂条件 2026调剂公告时间
    公务员调剂条件 2026调剂公告时间

    (一)符合拟调剂职位所要求的资格条件。 (二)公共科目笔试成绩同时达到拟调剂职位和原报考职位的合格分数线,且考试类别相同。 拟调剂职位设置了专业科目笔试条件的,专业科目笔试成绩还须同时达到合格分数线,且考试类别相同。 (三)未进入原报考职位面试人员名单。

    52

    2026.01.15

    国考成绩查询入口 国考分数公布时间2026
    国考成绩查询入口 国考分数公布时间2026

    笔试成绩查询入口已开通,考生可登录国家公务员局中央机关及其直属机构2026年度考试录用公务员专题网站http://bm.scs.gov.cn/pp/gkweb/core/web/ui/business/examResult/written_result.html,查询笔试成绩和合格分数线,点击“笔试成绩查询”按钮,凭借身份证及准考证进行查询。

    10

    2026.01.15

    Java 桌面应用开发(JavaFX 实战)
    Java 桌面应用开发(JavaFX 实战)

    本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

    65

    2026.01.14

    php与html混编教程大全
    php与html混编教程大全

    本专题整合了php和html混编相关教程,阅读专题下面的文章了解更多详细内容。

    36

    2026.01.13

    PHP 高性能
    PHP 高性能

    本专题整合了PHP高性能相关教程大全,阅读专题下面的文章了解更多详细内容。

    75

    2026.01.13

    热门下载

    更多
    网站特效
    /
    网站源码
    /
    网站素材
    /
    前端模板

    精品课程

    更多
    相关推荐
    /
    热门推荐
    /
    最新课程
    WEB前端教程【HTML5+CSS3+JS】
    WEB前端教程【HTML5+CSS3+JS】

    共101课时 | 8.3万人学习

    JS进阶与BootStrap学习
    JS进阶与BootStrap学习

    共39课时 | 3.2万人学习

    关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
    php中文网:公益在线php培训,帮助PHP学习者快速成长!
    关注服务号 技术交流群
    PHP中文网订阅号
    每天精选资源文章推送

    Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号