踩坑 规定某一节点展开,配置好逻辑无错误却失效
 <el-tree
      :data="menus"
      :props="defaultProps"
      show-checkbox="true"
      node-key="catId"
      :default-expand-all="false"
      :expand-on-click-node="false"
      :default-expanded-keys="[2, 3]"
     
    >
      <!-- node: 当前节点(包括节点的各种属性)  data:后台传递的节点数据 -->
      <span class="custom-tree-node" slot-scope="{ node, data }">
        <span>{{ node.label }}</span>
        <span>
          <el-button v-if="node.level <= 2" type="text" size="mini" @click="() => append(data)">增加</el-button>
          <el-button
            v-if="node.childNodes.length == 0"
            type="text"
            size="mini"
            @click="() => remove(node, data)"
          >删除</el-button>
        </span>
      </span>
    </el-tree>
需要展示哪一个节点,需要根据node-id找到该节点,在生成表单的时候,根据data对应的某一属性值进行赋值。
错误写法
<el-tree
      :data="menus"
      :props="defaultProps"
      show-checkbox="true"
      :node-key="catId"  我 加了 冒号“:” ,表明使用v-bind指令进行赋值,而我却使用的变量catId,而我在return data中却没有这个catId
      :default-expand-all="false"
      :expand-on-click-node="false"
      :default-expanded-keys="[2, 3]"
     
    >
我 加了 冒号“:” ,表明使用v-bind指令进行赋值。我使用的变量catId,却没有在return data中声明。这样Element-UI就不会使用:data=“menus” 中menus对象中对应的catId属性值了。导致node-key没有赋值,进而default-expanded-keys不能使用
正确做法,就是node-key 参数不要加冒号,默认使用data变量中的属性的值。
Element-ui 插件的使用
<!--  -->
<template>
  <div>
    <el-switch v-model="draggable" active-text="开启拖拽" inactive-text="关闭拖拽"></el-switch>
    <el-button v-if="draggable" type="primary" v-on:click="batchSave" round>批量保存</el-button>
    <el-button type="danger" @click="batchDelete">批量删除</el-button>
    <el-tree
      :data="menus"   菜单显示的数据
      :props="defaultProps" 对应参数的值。子节点的变量名,节点显示的变量名
      show-checkbox="true" 是否显示选择框
      node-key="catId" 选中菜单的id
      :default-expand-all="false" 菜单不展开,只显示一级菜单
      :expand-on-click-node="false" 当点击菜单的时候,不需要展开子菜单。只有点击箭头的时候,才会显示子菜单
      :default-expanded-keys="expandedkeys"  默认展开的节点的 key 的数组
      :draggable="draggable"   是否开启拖拽节点功能,true开启
      :allow-drop="allowDrop"   拖拽时判定目标节点能否被放置,使用allowDrop方法进行判断
      @node-drop="handleDrop"  拖拽成功完成时触发的事件
      ref="menuTree"   声明组件名字ref="名字"
    >
      <!--使用卡槽机制slot-scope,获取节点的信息。node: 当前节点(包括节点的各种属性值,以及填充的后台数据)  data:后台传递的节点数据 -->
      <span class="custom-tree-node" slot-scope="{ node, data }">
        <span>{{ node.label }}</span>
        <span>
          <!--一级和二级菜单才有增加按钮,没有子菜单的才可以删除,所有菜单都可以修改-->
          <el-button v-if="node.level <= 2" type="text" size="mini" @click="() => append(data)">增加</el-button>
          <!-- 也可以这样写事件: @click="edit(data)  或者  @click="() => append(data) -->
          <el-button type="text" size="mini" @click="edit(data)">修改</el-button>
          <el-button
            v-if="node.childNodes.length == 0"
            type="text"
            size="mini"
            @click="() => remove(node, data)"
          >删除</el-button>
        </span>
      </span>
    </el-tree>
    <!--弹框  -->
    <el-dialog title="提示" :visible.sync="dialogVisible" width="40%" :close-on-click-modal="false">
      <el-form :model="category">
        <el-form-item label="分类名称">
          <el-input v-model="category.name" autocomplete="off"></el-input>
        </el-form-item>
        <el-form-item label="图标">
          <el-input v-model="category.icon" autocomplete="off"></el-input>
        </el-form-item>
        <el-form-item label="计量单位">
          <el-input v-model="category.productUnit" autocomplete="off"></el-input>
        </el-form-item>
      </el-form>
      <span slot="footer" class="dialog-footer">
        <el-button @click="dialogVisible = false">取 消</el-button>
        <el-button type="primary" @click="submitData">确 定</el-button>
      </span>
    </el-dialog>
  </div>
</template>
<script>
export default {
  data() {
    return {
      pCid: [],
      updateNodes: [],
      maxLevel: 0,
      draggable: false,
      dialogType: "", //区分添加弹框add还是修改弹框edit
      title: "", //针对不同的弹框显示不同的标题
      category: {
        name: "",
        parentCid: 0,
        catLevel: 0,
        showStatus: 1,
        sort: 0,
        icon: "",
        productUnit: "",
      },
      dialogVisible: false, //为true时候显示弹框
      menus: [],
      expandedkeys: [],
      defaultProps: {
        children: "childrens", //子节点的变量名
        label: "name", //节点显示的名称变量名,与后台变量名称一致
      },
    };
  },
  methods: {
    // 获取数据列表
    getMenus() {
      this.dataListLoading = true;
      this.$http({
        url: this.$http.adornUrl("/product/category/list/tree"),
        method: "get", //get请求传递数据的时候,默认合并默认的参数值。起封装类在src/utils/httpRequest.js中。
      }).then(({ data }) => {
        console.log("成功了!!", data.data);
        this.menus = data.data;
      });
    },
    //批量删除分类
    batchDelete() {
      //使用组件自带的方法,声明组件名字ref="名字",this.$refs.名字.方法
      let checkedNodes = this.$refs.menuTree.getCheckedNodes(); //返回选中的节点的data数据
      let catIds = [];
      for(let i=0; i<checkedNodes.length; i++){
        let catId = checkedNodes[i].catId;
        catIds.push(catId);
      }
      this.$confirm(`是否将【${catIds}】分类删除`, "提示", {
        confirmButtonText: "确定",
        cancelButtonText: "取消",
        type: "warning",
      })
        .then(() => {
          //确定按钮
          //封装好了的http方法。
          this.$http({
            url: this.$http.adornUrl("/product/category/delete"),
            method: "post",
            data: this.$http.adornData(catIds, false), //adornData(传递的数据, false:表示不合并默认数据)
          }).then(({ data }) => {
            //发送请求之后调用成功
            this.$message({
              type: "success",
              message: "分类批量删除成功!",
            });
            //更新一下页面,并展示删除节点的父节点
            this.getMenus();
          });
        })
        .catch(() => {
          //取消按钮
          this.$message({
            type: "info",
            message: "已取消批量删除",
          });
        });
    },
    //拖拽成功的时候触发的事件handleDrop(拖拽节点,目的几点,相对于目的节点的位置,ev)
    handleDrop(draggingNode, dropNode, dropType, ev) {
      console.log("tree drop: ", draggingNode, dropNode, dropType, ev);
      //创建传递后台的对象,并将移动的每一个节点的数据封装成数组,传递到后台。节点移动后parentCid,catLevel,sort 其余的都不会改变
      //1. 获取父类id
      let pCid = 0; //更新后父id
      let siblings = null; //拖拽完成之后,所在父节点的所有子节点
      if (dropType == "before" || dropType == "after") {
        //考虑一级节点没父节点的情况
        pCid =
          dropNode.parent.data.catId == undefined
            ? 0
            : dropNode.parent.data.catId;
        siblings = dropNode.parent.childNodes;
      } else {
        pCid = dropNode.data.catId;
        siblings = dropNode.childNodes;
      }
      this.pCid.push(pCid); //将修改的节点的父节点存起来,用来保存成功后的展示
      //2.获取最新排序和层级(遍历拖拽节点的父节点所有子节点)
      for (let i = 0; i < siblings.length; i++) {
        //node几点信息随着节点的移动,其属性所在的层级随之发生改变,但是数据库中的信息不会发生改变。
        //所以只要将数据库中的层级catLevel与node节点level同步信息就可以。
        if (siblings[i].data.catId == draggingNode.data.catId) {
          //遍历到拖拽节点
          let catLevel = draggingNode.level;
          if (siblings[i].level != draggingNode.level) {
            //移动后层级发生变化
            catLevel = siblings[i].level;
            //其所有的子节点也要同步
            this.updateChildNodeLevel(siblings[i]);
          }
          //传递到后台的对象
          this.updateNodes.push({
            catId: siblings[i].data.catId,
            parentCid: pCid,
            sort: i,
            catLevel: catLevel,
          });
        } else {
          this.updateNodes.push({
            catId: siblings[i].data.catId,
            sort: i,
          });
        }
      }
      console.log("提交后台的数据:", this.updateNodes);
    },
    //  同步其子节点的层级数据
    updateChildNodeLevel(node) {
      if (node.childNodes != null && node.childNodes.length > 0) {
        for (let i = 0; i < node.childNodes.length; i++) {
          //子节点只是变了层级,父节点id和sort都没变。
          this.updateNodes.push({
            catId: node.childNodes[i].data.catId,
            catLevel: node.childNodes[i].level,
          });
          this.updateChildNodeLevel(node.childNodes[i]);
        }
      }
    },
    //批量保存
    batchSave() {
      //首先 创建传递后台的对象,节点移动后parentCid,catLevel,sort 其余的都不会改变。
      //思路:将拖拽的每一个节点所更新的数据保存到一个数组中,最后批量保存将数组传递到后台。
      // 需要一个在拖拽成功的时候,触发的一个事件。node-drop 事件
      this.$http({
        url: this.$http.adornUrl("/product/category/update/batchSave"),
        method: "post",
        data: this.$http.adornData(this.updateNodes, false),
      }).then(({ data }) => {
        this.$message({
          type: "success",
          message: "批量保存成功!",
        });
        //更新一下页面,并展示新增节点的父节点
        this.getMenus();
        this.expandedkeys = this.pCid;
        //这地方一定要置为空!!!!
        this.updateNodes = [];
        this.maxLevel = 0;
      });
    },
    //进行菜单拖拽功能的判定
    allowDrop(draggingNode, dropNode, type) {
      //被拖动的当前节点以及所在的父节点总层数不能大于3
      //1)、被拖动的当前节点总层数
      console.log("allowDrop:", draggingNode, dropNode, type);
      //递归查询该节点最大深度
      this.countNodeLevel(draggingNode);
      console.log("最大深度:", this.maxLevel);
      // //当前正在拖动的节点+父节点所在的深度不大于3即可
      let deep = Math.abs(this.maxLevel - draggingNode.level) + 1; //以该节点为起点的层数
      //这地方一定要置为0,否则将会出现错误!!!!大坑!!!!!!!!!
      this.maxLevel = 0;
      if (type == "inner") {
        // console.log(
        //   `this.maxLevel:${this.maxLevel};draggingNode.data.catLevel:${draggingNode.data.catLevel};dropNode.level:${dropNode.level}`
        // );
        return deep + dropNode.level <= 3;
      } else {
        return deep + dropNode.parent.level <= 3;
      }
    },
    //找到子节点最大的深度
    countNodeLevel(node) {
      //找到所有子节点,求出最大深度(node中有一个变量参数为data,是数据库返回的数据。其余的参数还有prent父节点,level层数,这个随着节点的拖拽进行改变,但是数据库中的数据不发生改变。)
      if (node.childNodes != null && node.childNodes.length > 0) {
        for (let i = 0; i < node.childNodes.length; i++) {
          if (node.childNodes[i].level > this.maxLevel) {
            this.maxLevel = node.childNodes[i].level;
          }
          this.countNodeLevel(node.childNodes[i]);
        }
      }
    },
    //添加节点,显示弹框
    append(data) {
      console.log("增加数据", data);
      this.category.name = "";
      this.category.parentCid = data.catId; //增加节点为该节点的子节点,将该节点id是其子节点的父id
      this.category.catLevel = data.catLevel * 1 + 1; //避免是字符串,先乘以1,转换成数字
      this.category.showStatus = 1;
      this.category.sort = 0;
      this.category.icon = "";
      this.category.productUnit = "";
      this.dialogType = "add";
      this.title = "添加分类";
      this.dialogVisible = true; //显示弹框
    },
    //修改节点,显示弹框
    edit(data) {
      console.log("要修改的数据:", data);
      this.dialogType = "edit";
      this.title = "修改分类";
      this.dialogVisible = true; //显示弹框
      //当一人在某一时间的页面修改之后,另在某一时间之前修改,要获取最新数据,重新向后台发送请求。
      this.$http({
        url: this.$http.adornUrl(`/product/category/info/${data.catId}`),
        method: "get",
      }).then(({ data }) => {
        //{data},如果不解构的话,返回的信息包括头信息,配置信息,以及后台数据R对象。解构之后直接获得的R对象,因为前台名字为{data:{R对象的json},....}
        //请求成功,要回显的数据
        console.log("回显数据:", data);
        this.category.catId = data.category.catId;
        this.category.name = data.category.name;
        this.category.parentCid = data.category.parentCid;
        this.category.catLevel = data.category.catLevel;
        this.category.showStatus = data.category.showStatus;
        this.category.sort = data.category.sort;
        this.category.icon = data.category.icon;
        this.category.productUnit = data.category.productUnit;
      });
    },
    //判断是添加还是修改
    submitData() {
      if (this.dialogType == "add") {
        this.addCategory();
      } else if (this.dialogType == "edit") {
        this.editCategory();
      }
    },
    //添加分类
    addCategory() {
      this.$http({
        url: this.$http.adornUrl("/product/category/save"),
        method: "post",
        data: this.$http.adornData(this.category, false),
      }).then(({ data }) => {
        this.$message({
          type: "success",
          message: "分类添加成功!",
        });
        //关闭弹框
        this.dialogVisible = false;
        //更新一下页面,并展示新增节点的父节点
        this.getMenus();
        this.expandedkeys = [this.category.parentCid];
      });
    },
    //修改分类
    editCategory() {
      //为了提高效率,将不修改的字段不需要传到后台,mybatis plus 将为空的字段sql语句会去掉。
      //使用es6结构方法
      var { catId, name, icon, productUnit } = this.category;
      this.$http({
        url: this.$http.adornUrl("/product/category/update"),
        method: "post",
        //this.$http.adornData({catId:catId, name:name, icon:icon, productUnit:productUnit}, false) es6写法==》
        data: this.$http.adornData({ catId, name, icon, productUnit }, false),
      }).then(({ data }) => {
        this.$message({
          type: "success",
          message: "分类修改成功!",
        });
        //关闭弹框
        this.dialogVisible = false;
        //更新一下页面,并展示修改节点的父节点
        this.getMenus();
        this.expandedkeys = [this.category.parentCid];
      });
    },
    //删除节点
    remove(node, data) {
      //发送给后端的数据,后端是数组类型
      var ids = [data.catId];
      this.$confirm(`是否将【${data.name}】删除`, "提示", {
        confirmButtonText: "确定",
        cancelButtonText: "取消",
        type: "warning",
      })
        .then(() => {
          //确定按钮
          //封装好了的http方法。
          this.$http({
            url: this.$http.adornUrl("/product/category/delete"),
            method: "post",
            data: this.$http.adornData(ids, false), //adornData(传递的数据, false:表示不合并默认数据)
          }).then(({ data }) => {
            //发送请求之后调用成功
            this.$message({
              type: "success",
              message: "分类删除成功!",
            });
            //更新一下页面,并展示删除节点的父节点
            this.getMenus();
            this.expandedkeys = [node.parent.data.catId];
          });
        })
        .catch(() => {
          //取消按钮
          this.$message({
            type: "info",
            message: "已取消删除",
          });
        });
    },
  },
  //周期函数 vue对象创建成功后调用
  created: function () {
    this.getMenus();
  },
};
</script>
<style>
</style>
对应的后台
package com.atguigu.gulimall.product.controller;
import java.util.Arrays;
import java.util.List;
import java.util.ListIterator;
import java.util.Map;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import com.atguigu.gulimall.product.entity.CategoryEntity;
import com.atguigu.gulimall.product.service.CategoryService;
import com.atguigu.common.utils.PageUtils;
import com.atguigu.common.utils.R;
/**
 * 商品三级分类
 *
 * @author hzm
 * @email hzmxiansheng@123.com
 * @date 2020-08-16 12:04:18
 */
@RestController
@RequestMapping("product/category")
public class CategoryController {
    @Autowired
    private CategoryService categoryService;
    /**
     * 查询所有分类以及子分类,以树形结构组装起来
     */
    @RequestMapping("/list/tree")
    public R list(){
        List<CategoryEntity> entities = categoryService.listWithTree();
        // R 继承了 HashMap ,是一个Map集合,返回的前端数据
        return R.ok().put("data", entities);
    }
    /**
     * 获取某一菜单的信息
     */
    @RequestMapping("/info/{catId}")
    public R info(@PathVariable("catId") Long catId){
		CategoryEntity category = categoryService.getById(catId);
        return R.ok().put("category", category);
    }
    /**
     * 菜单的保存
     */
    @RequestMapping("/save")
    public R save(@RequestBody CategoryEntity category){
		categoryService.save(category);
        return R.ok();
    }
    /**
     * 菜单的修改
     */
    @RequestMapping("/update")
    public R update(@RequestBody CategoryEntity category){
		categoryService.updateById(category);
        return R.ok();
    }
    /**
     * 菜单的批量保存
     */
    @RequestMapping("/update/batchSave")
    public R updateBatchSave(@RequestBody CategoryEntity[] categorys){
        //菜单的批量修改
		categoryService.updateBatchById(Arrays.asList(categorys));
        return R.ok();
    }
    /**
     * 菜单的删除(逻辑删除)
     */
    @RequestMapping("/delete")
    public R delete(@RequestBody Long[] catIds){
//		categoryService.removeByIds(Arrays.asList(catIds)); // 批量删除
        categoryService.removeMenusByIds(Arrays.asList(catIds));
        return R.ok();
    }
}
效果图
Mybatis Plus 框架的逻辑删除
1)、配置全局的逻辑删除规则(省略)
 在application.yml文件中配置逻辑删除的规则
mybatis-plus:
  mapper-locations: classpath:/mapper/**/*.xml
  global-config:
    db-config:
      id-type: auto  # 主键自增
      logic-delete-field: flag  # 全局逻辑删除的实体字段名(since 3.3.0,配置后可以忽略不配置步骤2)
      logic-delete-value: 1 # 逻辑已删除值(默认为 1)
      logic-not-delete-value: 0 # 逻辑未删除值(默认为 0) 如果逻辑删除的值是反了的话,在表对应的javabean中使用注解@TableLogic进行设置自己的逻辑
2)、配置逻辑删除的组件Bean(省略)
 这个在Mybatis-plus版本3.1之后就省略了。
3)、给Bean加上逻辑删除注解@TableLogic
 在javaBean中对相应的字段张添加注解,并且可以修改逻辑删除的规则。
//对应的CategoryEntity封装类
@TableLogic(value = "1",delval = "0")
	private Integer showStatus;
版权声明:本文为weixin_44142032原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。