0

0

Svelte 与 SortableJS 集成:正确处理动态嵌套数组的拖拽排序

碧海醫心

碧海醫心

发布时间:2025-12-29 21:07:26

|

874人浏览过

|

来源于php中文网

原创

Svelte 与 SortableJS 集成:正确处理动态嵌套数组的拖拽排序

本文详解如何在 svelte 中结合 sortablejs 实现多列表(嵌套数组)的稳定拖拽排序,重点解决因 `#each` 缺失 key 导致的 ui 错乱、状态不同步及双渲染问题,并提供基于 action 的简洁、可维护实现方案。

在 Svelte 中集成 SortableJS 实现跨列表拖拽时,常见“抖动”“回跳”“重复移动”等异常行为,根本原因往往不是 SortableJS 本身,而是 Svelte 的响应式更新机制与 DOM 状态未对齐。最典型的问题是:{#each} 块缺少唯一 key 表达式,导致 Svelte 无法准确追踪每个

  • 元素的身份,从而在排序后错误复用或销毁节点,引发视觉错乱和逻辑混乱。

    ✅ 正确做法:始终为 {#each category as item} 添加 key —— 使用 item.id(或其他稳定唯一标识):

    {#each category as item (item.id)}
      <li data-id={item.id}>{item.name}</li>
    {/each}

    不加 (item.id) 会导致 Svelte 按索引位置比对元素,而拖拽会改变索引顺序,造成 DOM 节点被错误移动或重渲染,进而干扰 SortableJS 的内部状态。

    更进一步,避免将 Sortable 初始化封装进独立组件(如 List.svelte)。自定义组件会引入额外的生命周期、作用域和响应式绑定复杂度,容易引发 fullArr[index] 赋值后视图未及时同步、onSort 多次触发、或 sortable.toArray() 返回旧 ID 序列等问题。

    Stable Diffusion 2.1 Demo
    Stable Diffusion 2.1 Demo

    最新体验版 Stable Diffusion 2.1

    下载

    推荐采用 Svelte Action(use: 指令) —— 它天然与 DOM 元素绑定,生命周期清晰(仅在元素挂载/卸载时执行),且逻辑集中、无状态泄漏风险。以下是生产就绪的实现:

    ✅ 推荐方案:使用 use: action + onEnd 精准更新(推荐用于跨列表拖拽)

    <script>
      import Sortable from 'sortablejs';
    
      let items = [
        [{ id: 1, name: "one" }, { id: 2, name: "two" }],
        [{ id: 3, name: "three" }, { id: 4, name: "four" }]
      ];
    
      function initSortable(node, listIndex) {
        const sortable = new Sortable(node, {
          group: 'shared-lists', // 同名 group 支持跨列表拖拽
          onEnd(event) {
            // 从源列表移除
            const fromItems = items[event.from.dataset.listIndex];
            const [movedItem] = fromItems.splice(event.oldIndex, 1);
    
            // 插入目标列表(注意:newIndex 是插入前的位置)
            const toItems = items[event.to.dataset.listIndex];
            toItems.splice(event.newIndex, 0, movedItem);
    
            // 强制触发响应式更新(因 .splice 修改原数组,Svelte 不自动检测)
            items = [...items];
          }
        });
        return {
          destroy() {
            sortable.destroy();
          }
        };
      }
    </script>
    
    {#each items as category, i}
      <h2>Category {i}</h2>
      <ul data-list-index={i} use:initSortable={i}>
        {#each category as item (item.id)}
          <li data-id={item.id}>{item.name}</li>
        {/each}
      </ul>
    {/each}
    
    <!-- 实时查看状态 -->
    <pre class="brush:php;toolbar:false;">{JSON.stringify(items, null, 2)}

    ? 关键要点:

    • data-list-index 将列表索引透传给 DOM,供 onEnd 中安全读取(避免闭包捕获过期 i);
    • 使用 onEnd 而非 onSort:onSort 在拖拽中高频触发,且 toArray() 可能返回中间态 ID;onEnd 仅在操作完成时调用,数据最终一致;
    • items = [...items] 是必需的:Svelte 仅对赋值操作(=)进行响应式追踪,.splice() 属于原地修改,需显式触发更新;
    • use: action 的 destroy 钩子确保组件卸载时清理 Sortable 实例,防止内存泄漏。

    ⚠️ 注意事项

    • 勿直接修改 items[i] = newArray:若 newArray 是新引用但元素相同,Svelte 可能跳过更新;应统一用 items = [...items] 或 items = structuredClone(items) 触发变更。
    • ID 必须全局唯一:跨列表拖拽依赖 item.id 查找,重复 ID 将导致匹配错误。
    • 避免在 onSort 中调用 fullArr.flat().find(...):该方式性能差(O(n²)),且在多列表场景下易因数组未及时更新而查到错误项;onEnd + 显式移动语义更可靠、高效。

    通过 key 化 #each 和 action 驱动的精准状态管理,即可彻底告别“抖动列表”,构建出响应迅速、逻辑清晰、易于扩展的 Svelte 多列表拖拽系统。

  • 热门AI工具

    更多
    DeepSeek
    DeepSeek

    幻方量化公司旗下的开源大模型平台

    豆包大模型
    豆包大模型

    字节跳动自主研发的一系列大型语言模型

    通义千问
    通义千问

    阿里巴巴推出的全能AI助手

    腾讯元宝
    腾讯元宝

    腾讯混元平台推出的AI助手

    文心一言
    文心一言

    文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

    讯飞写作
    讯飞写作

    基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

    即梦AI
    即梦AI

    一站式AI创作平台,免费AI图片和视频生成。

    ChatGPT
    ChatGPT

    最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

    相关专题

    更多
    go语言闭包相关教程大全
    go语言闭包相关教程大全

    本专题整合了go语言闭包相关数据,阅读专题下面的文章了解更多相关内容。

    151

    2025.07.29

    DOM是什么意思
    DOM是什么意思

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

    4283

    2024.08.14

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

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

    436

    2023.08.03

    JavaScript浏览器渲染机制与前端性能优化实践
    JavaScript浏览器渲染机制与前端性能优化实践

    本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

    58

    2026.03.06

    Rust内存安全机制与所有权模型深度实践
    Rust内存安全机制与所有权模型深度实践

    本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

    146

    2026.03.05

    PHP高性能API设计与Laravel服务架构实践
    PHP高性能API设计与Laravel服务架构实践

    本专题围绕 PHP 在现代 Web 后端开发中的高性能实践展开,重点讲解基于 Laravel 框架构建可扩展 API 服务的核心方法。内容涵盖路由与中间件机制、服务容器与依赖注入、接口版本管理、缓存策略设计以及队列异步处理方案。同时结合高并发场景,深入分析性能瓶颈定位与优化思路,帮助开发者构建稳定、高效、易维护的 PHP 后端服务体系。

    273

    2026.03.04

    AI安装教程大全
    AI安装教程大全

    2026最全AI工具安装教程专题:包含各版本AI绘图、AI视频、智能办公软件的本地化部署手册。全篇零基础友好,附带最新模型下载地址、一键安装脚本及常见报错修复方案。每日更新,收藏这一篇就够了,让AI安装不再报错!

    93

    2026.03.04

    Swift iOS架构设计与MVVM模式实战
    Swift iOS架构设计与MVVM模式实战

    本专题聚焦 Swift 在 iOS 应用架构设计中的实践,系统讲解 MVVM 模式的核心思想、数据绑定机制、模块拆分策略以及组件化开发方法。内容涵盖网络层封装、状态管理、依赖注入与性能优化技巧。通过完整项目案例,帮助开发者构建结构清晰、可维护性强的 iOS 应用架构体系。

    159

    2026.03.03

    C++高性能网络编程与Reactor模型实践
    C++高性能网络编程与Reactor模型实践

    本专题围绕 C++ 在高性能网络服务开发中的应用展开,深入讲解 Socket 编程、多路复用机制、Reactor 模型设计原理以及线程池协作策略。内容涵盖 epoll 实现机制、内存管理优化、连接管理策略与高并发场景下的性能调优方法。通过构建高并发网络服务器实战案例,帮助开发者掌握 C++ 在底层系统与网络通信领域的核心技术。

    30

    2026.03.03

    热门下载

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

    精品课程

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

    共101课时 | 10.1万人学习

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

    共39课时 | 3.3万人学习

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

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