[TOC]⼀般来说后端会提供接⼝将当前章节最新顺序上传,但项⽬中没有提供这样的接⼝,提供的是单个课时位置更新的接⼝,所以我们需要遍历列表,依次更新处理(好处是可以练习批量请求的处理操作)。 ⾸先封装相关接⼝:⽤于更新章节信息的 saveOrUpdateSection 接⼝。 ⽤于更新课时的 saveOrUpdateLesson 接⼝(位于⽂档的课时内容分类下)。 // services/course-section.js … // 新增或更新章节 export const saveOrUpdateSection = data => { return request({ method: ‘POST’, url: ‘/boss/course/section/saveOrUpdateSection’, data }) } // 新增或更新课时(因课时功能较少,此处未单独封装模块,可⾃⾏处理) export const saveOrUpdateLesson = data => { return request({ method: ‘POST’, url: ‘/boss/course/lesson/saveOrUpdate’, data }) } Tree 组件提供了 node-drop ⽅法(事件)⽤于设置拖动后的处理。// section.vue … … @node-drop=“handleNodeDrop” > … import { getSectionAndLesson, saveOrUpdateSection, saveOrUpdateLe sson } from ‘@/services/course-section.js’ … // 设置节点拖动后的数据更新 async handleNodeDrop (draggingNode, dropNode, tyoe, event) { // 1. ⽆论是章节还是课时, dropNode 都有parent(draggingNode.parent 总为 null), 内部有childNodes // - dropNode.parent.childNodes 可获取拖拽项所在列表的所有数据 // - 遍历操作 // 4. 由于是批量请求,可以使⽤ Promise.all() 便于进⾏统⼀操作 // - 将 map 返回的,由 Axios 调⽤返回的 Promise 对象组成的数组,传⼊ 到 Promise.all() 中 // - 设置 async await 并进⾏ try..catch 处理 try { await Promise.all(dropNode.parent.childNodes.map((item, index ) => { // 2. 对章节与课时进⾏分别处理 // - 除了 draggingNode.data.sectionId 外,draggingNode.les sonDTOS 也可以判断 if (draggingNode.data.lessonDTOS) { // 章节操作 return saveOrUpdateSection({ id: item.data.id, // 按现在的索引顺序给当前级别列表设置排序序号 orderNum: index }) } else { // 课时操作(同上) return saveOrUpdateLesson({ id: item.data.id, // 按现在的索引顺序给当前级别列表设置排序序号 orderNum: index }) } })) this.$message.success(‘数据更新成功’) } catch (err) { this.$message.success(‘数据更新失败’, err) } } … … 设置完毕,可以在请求过程中添加 loading 效果,以体会 Promise.all() 的好处。示例中为“遍历+请求”结构,这时如果设置 await 就会变成⼀个⼀个顺序发送请求。 通过 Promise.all() 可以统⼀发送,并统⼀进⾏结束处理,⼗分⽅便。 // course-section.vue … v-loading=“isLoading” … > … data () { return { … isLoading: false } }, … async handleNodeDrop (draggingNode, dropNode, tyoe, event) { this.isLoading = true … try { … } catch (err) { … } this.isLoading = false } 完成。此部分主要为业务练习,实际开发时让后端提供对应接⼝即可快速实现了。