商品服务 - 三级分类

1.递归查询树形结构

 @Override
    public List<CategoryEntity> listWithTree() {
        //1.查出所有分类

        List<CategoryEntity> all = this.list();
        //2.组装成父子的属性结构
        List<CategoryEntity> level1Menus = all
                .stream()
                .filter(c -> c.getParentCid().equals(0L))
                .map(categoryEntity -> categoryEntity.setChildren(getChildrenCategory(all,categoryEntity.getCatId())))
                //大于放后面 升序
                .sorted(Comparator.comparing(CategoryEntity::getSort))
                .collect(Collectors.toList());

        return level1Menus;
    }

    private List<CategoryEntity> getChildrenCategory(List<CategoryEntity> allList, long pCatId) {
        List<CategoryEntity> collect = allList.stream()
                .filter(a -> a.getParentCid() == pCatId)
                .sorted(Comparator.comparing(CategoryEntity::getSort))
                .collect(Collectors.toList());
        if (collect.isEmpty()) {
            return new ArrayList<>();
        } else {
            for (CategoryEntity categoryEntity : collect) {
                Long catId = categoryEntity.getCatId();
                List<CategoryEntity> childrenCategory = getChildrenCategory(allList, catId);
                categoryEntity.setChildren(childrenCategory);
            }
            return collect;
        }
    }

2.网关统一配置跨域问题

 

2.1添加 网关过滤器

package com.jmj.gulimall.gateway.config;

import org.springframework.context.annotation.Bean;
import org.springframework.web.cors.CorsConfiguration;
import org.springframework.web.cors.reactive.UrlBasedCorsConfigurationSource;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.cors.reactive.CorsWebFilter;


@Configuration
public class GulimallCorsConfiguration {

    //网关gateway是使用webflex 进行编程的 响应式 所以用的都是reactive 包下
    @Bean
    public CorsWebFilter corsWebFilter() {

        UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource();
        CorsConfiguration corsConfiguration = new CorsConfiguration();
        //1.配置跨域
        corsConfiguration.addAllowedHeader("*");//允许哪些头跨域
        corsConfiguration.addAllowedMethod("*");//允许哪些请求跨域
        corsConfiguration.addAllowedOrigin("*");//允许哪些请求来源 跨域
        corsConfiguration.setAllowCredentials(true);//是否允许携带cookie进行跨域 允许 否则跨域请求将会丢失一些相关cookie信息

        source.registerCorsConfiguration("/**", corsConfiguration);
        return new CorsWebFilter(source);

    }


}

 但是我们发现还是出现了问题

这是因为网关配置了跨域,而网关转发的微服务也配置了跨域,所以返回了两个响应头被允许,

但是浏览器只希望有一个,所以报错,这时,我们只需要去把微服务的跨域注释掉就好了。 

这是人人fast 服务的跨域配置,也是Spring MVC的 配置

/**
 * Copyright (c) 2016-2019 人人开源 All rights reserved.
 * <p>
 * https://www.renren.io
 * <p>
 * 版权所有,侵权必究!
 */

package io.renren.config;

import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.CorsRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;

@Configuration
public class CorsConfig implements WebMvcConfigurer {

    @Override
    public void addCorsMappings(CorsRegistry registry) {
        registry.addMapping("/**")
                .allowedOrigins("*")
                .allowCredentials(true)
                .allowedMethods("GET", "POST", "PUT", "DELETE", "OPTIONS")
                .maxAge(3600);
    }
}

 

这是网关的路由,是有优先级的,从上优先级最高,如果匹配不上就会依次遍历。

如果不调换优先级,路径会被网关转发到 renrenfast 服务当中,导致没有token 返回invalid token  

spring:
  cloud:
    gateway:
      routes:
        - id: test_route
          uri: http://www.baidu.com
          predicates:
            #访问的路径就是 如果是/hello?url=baidu  就转发到 https://www.baidu.com/hello?url=baidu
            - Query=url,baidu
        - id: test1_route
          uri: http://www.hao123.com
          predicates:
            - Query=url,hao123
        - id: product_rout
          uri: lb://gulimall-product
          predicates:
            - Path=/api/product/**
          filters:
            - RewritePath=/api/(?<segment>.*),/$\{segment}
        - id: admin_route
          uri: lb://renren-fast
          predicates:
            - Path=/api/**
          filters:
            - RewritePath=/api/(?<segment>.*),/renren-fast/$\{segment}


##前端项目,/api

3.拖拽修改前端代码

<!--  -->
<template>
  <div>
    <el-switch v-model="isDraggable" active-text="开启拖拽" inactive-text="关闭拖拽">
    </el-switch>
    <el-tree
      :data="menus"
      :props="defaultProps"
      show-checkbox
      node-key="catId"
      :expand-on-click-node="false"
      :default-expanded-keys="expandedkey"
      @node-expand="expend"
      @node-collapse="nodeClose"
      :draggable="isDraggable"
      :allow-drop="allowDrop"
      @node-drop="handleDrop"
    >
      <span class="custom-tree-node" slot-scope="{ node, data }">
        <!-- 插槽,代替了原来的 label  里所显示的内容  将插槽内容显示在原来的每个结点上面  -->
        <span>{{ node.label }}</span>
        <span>
          <el-button
            v-if="node.level <= 2"
            type="text"
            size="mini"
            @click="() => append(data)"
          >
            Append
          </el-button>
          <el-button type="text" size="mini" @click="edit(data)">
            Edit
          </el-button>

          <el-button
            v-if="node.childNodes.length == 0"
            type="text"
            size="mini"
            @click="() => remove(node, data)"
          >
            Delete
          </el-button>
        </span>
      </span>
    </el-tree>

    <el-dialog
      :title="submitTitle"
      :visible.sync="dialogVisible"
      :close-on-click-modal="false"
      width="30%"
    >
      <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>
      <div slot="footer" class="dialog-footer">
        <el-button @click="cancelForm">取 消</el-button>
        <el-button type="primary" @click="submitForm">确 定</el-button>
      </div>
    </el-dialog>
  </div>
</template>

<script>
//这里可以导入其他文件(比如:组件,工具js,第三方插件js,json文件,图片文件等等)
//例如:import 《组件名称》 from '《组件路径》';
class UpdateCategoryData {
  constructor(catId, parentCid, sort, catLevel) {
    this.catId = catId;
    this.parentCid = parentCid;
    this.sort = sort;
    this.catLevel = catLevel;
  }
}
export default {
  //import引入的组件需要注入到对象中才能使用
  components: {},
  props: {},
  data() {
    return {
      isDraggable: false,
      submitType: "",
      submitTitle: "",
      dialogVisible: false,
      menus: [],
      defaultProps: {
        children: "children",
        label: "name",
      },
      expandedkey: [],
      copyCategory: {},
      category: {
        catId: null,
        name: "",
        parentCid: 0,
        catLevel: 0,
        showStatus: 1,
        productUnit: "",
        icon: "",
        sort: 0,
      },
    };
  },
  //监听属性 类似于data概念
  computed: {},
  //监控data中的数据变化
  watch: {},
  //方法集合
  methods: {
    /**
     *
     * @param {*} draggingNode 拖拽的节点
     * @param {*} dropNode  拖拽到的哪个节点
     * @param {*} dropType 拖拽类型, 前后或者 内部
     * @param {*} event  事件对象
     */
    handleDrop(draggingNode, dropNode, dropType, event) {
      console.log(draggingNode, dropNode, dropType);

      //1.当前节点最新的父节点Id,
      let updateArray = new Array();

      let draggingNodeLevel = draggingNode.level;
      if (dropType == "inner") {
        let dropNodeId = dropNode.data.catId;
        let dropNodeLevel = dropNode.level;
        let childrenNew = dropNode.data.children;
        for (const index in childrenNew) {
          let { catId } = childrenNew[index];
          let updateCategoryData = new UpdateCategoryData(
            catId,
            dropNodeId,
            index,
            dropNodeLevel + 1
          );
          updateArray.push(updateCategoryData);
        }

        let div = dropNodeLevel + 1 - draggingNodeLevel;
        //递归把子节点都遍历完
        if (div != 0) {
          this.recursivelyTraverseChildNodes(
            draggingNode.data.children,
            div,
            updateArray
          );
        }
      } else {
        //往前插入节点或者后插入节点
        let parentLevel = dropNode.parent.level;
        console.log(parentLevel);
        let parentChildrenArr = {};
        if (parentLevel == 0) {
          parentChildrenArr = dropNode.parent.data;
        } else {
          parentChildrenArr = dropNode.parent.data.children;
        }

        let parentCid = dropNode.data.parentCid;
        for (const index in parentChildrenArr) {
          let { catId } = parentChildrenArr[index];
          let updateCategoryData = new UpdateCategoryData(
            catId,
            parentCid,
            index,
            parentLevel + 1
          );
          updateArray.push(updateCategoryData);
        }

        let div = parentLevel + 1 - draggingNodeLevel;
        console.log("parentLevel", parentLevel);
        console.log("draggingNodeLevel", draggingNodeLevel);
        //递归把子节点都遍历完
        if (div != 0) {
          this.recursivelyTraverseChildNodes(
            draggingNode.data.children,
            div,
            updateArray
          );
        }
      }

      console.log(updateArray);
      //发送http请求修改
      this.$http({
        url: this.$http.adornUrl("/product/category/updateList"),
        method: "post",
        data: this.$http.adornData(updateArray, false),
      }).then(({ data }) => {
        this.success("修改位置与排序成功");
        this.getMenus();
      });
    },
    recursivelyTraverseChildNodes(children, div, arr) {
      if (children == null || children.length == 0) {
        //没有子节点了
        return;
      } else {
        for (const child of children) {
          let updateCategoryData = new UpdateCategoryData(
            child.catId,
            child.parentCid,
            child.sort,
            child.catLevel + div
          );
          arr.push(updateCategoryData);
          this.recursivelyTraverseChildNodes(child.children, div, arr);
        }
      }
    },
    allowDrop(draggingNode, dropNode, type) {
      //1、被拖动的当前节点以及所在的父节点总层数不能大于3
      console.log("allowDrop", draggingNode, dropNode, type);
      let level = this.countNodeLevel(
        draggingNode.data,
        draggingNode.data.catLevel
      );
      if (type == "inner") {
        if (dropNode.level + level <= 3) {
          return true;
        } else {
          return false;
        }
      }

      if (type == "next") {
        if (dropNode.parent.level + level <= 3) {
          return true;
        } else {
          return false;
        }
      }

      if (type == "prev") {
        if (dropNode.parent.level + level <= 3) {
          return true;
        } else {
          return false;
        }
      }
      //1)被拖动的当前节点总层数
      return false;
    },
    //递归查找该节点加上子类最深层级一共有几层;包含自己算
    countNodeLevel(node, l) {
      let children = node.children;
      let levelMax = 0;
      if (children.length == 0) {
        return node.catLevel - l + 1;
      } else {
        for (let child of children) {
          let level = this.countNodeLevel(child, l);
          if (level > levelMax) {
            levelMax = level;
          }
        }
      }
      return levelMax;
    },
    resetCategory() {
      Object.assign(this.category, this.copyCategory);
      //如果你希望在 console.log 输出的时候看到对象的当前状态,
      //你可以在赋值操作之前进行 console.log,或者使用对象解构等方法创建一个新的对象进行输出,以确保输出的是当前状态的副本而不是对象的引用。
      let categoryre = {};
      Object.assign(categoryre, this.category);
      console.log("执行了重置", categoryre);
    },
    submitForm() {
      if (this.submitType == "add") {
        this.addCategory();
      }
      if (this.submitType == "edit") {
        this.updateCategory();
      }
    },
    updateCategory() {
      this.$http({
        url: this.$http.adornUrl("/product/category/update"),
        method: "post",
        data: this.$http.adornData(this.category, false),
      }).then(({ data }) => {
        this.success("修改成功");
        this.getMenus();
        this.resetCategory();
        this.dialogVisible = false;
      });
    },
    edit(data) {
      this.submitType = "edit";
      this.submitTitle = "修改分类菜单";
      console.log("正在修改数据", data);
      this.dialogVisible = true;

      //发送http请求获取回显数据
      this.$http({
        url: this.$http.adornUrl(`/product/category/info/${data.catId}`),
        method: "get",
      }).then(({ data }) => {
        let categoryEdit = data.data;
        console.log("回显数据", categoryEdit);
        this.category.catId = categoryEdit.catId;
        this.category.name = categoryEdit.name;
        this.category.parentCid = categoryEdit.parentCid;
        this.category.catLevel = categoryEdit.catLevel;
        this.category.showStatus = categoryEdit.showStatus;
        this.category.productUnit = categoryEdit.productUnit;
        this.category.icon = categoryEdit.icon;
        this.category.sort = categoryEdit.sort;
        console.log("category被回显数据", this.category);
      });
    },
    cancelForm() {
      this.resetCategory();
      this.dialogVisible = false;
    },
    append(data) {
      this.resetCategory();
      console.log("append", this.category);
      this.category.parentCid = data.catId;
      this.category.catLevel = data.catLevel + 1;
      this.category.showStatus = 1;
      this.category.sort = 0;
      this.dialogVisible = true;
      this.submitType = "add";
      this.submitTitle = "添加分类菜单";
    },
    addCategory() {
      console.log("提交三级分类的数据", this.category);

      this.$http({
        url: this.$http.adornUrl("/product/category/save"),
        method: "post",
        data: this.$http.adornData(this.category, false),
      }).then(({ data }) => {
        this.success("添加分类成功");
        this.getMenus();
      });

      this.resetCategory();
      this.dialogVisible = false;
    },
    remove(node, data) {
      console.log("remove", node, data);
      let ids = [data.catId];
      // console.log(this); //vue
      this.$confirm(`是否删除【${data.name}】菜单`, "提示", {
        confirmButtonText: "确定",
        cancelButtonText: "取消",
        type: "warning",
      })
        .then(() => {
          // console.log(this); //vue
          this.$http({
            url: this.$http.adornUrl("/product/category/delete"),
            method: "post",
            data: this.$http.adornData(ids, false),
          }).then(({ data }) => {
            // console.log(this); //vue
            this.$message({
              message: "菜单删除成功",
              type: "success",
            });
            // node.visible = false;
            this.getMenus();
            //设置需要默认展开的菜单
          });
        })
        .catch(() => {
          this.$message({
            message: "取消了删除",
            type: "warning",
          });
        });
    },
    expend(data, node, _) {
      console.log("展开了", node.data.catId);
      this.expandedkey.push(node.data.catId);
    },
    nodeClose(data, node, _) {
      let id = node.data.catId;
      console.log("收起了", id);
      let index = this.expandedkey.indexOf(id);
      this.expandedkey.splice(index, 1);
    },
    getMenus() {
      this.$http({
        url: this.$http.adornUrl("/product/category/list/tree"),
        method: "get",
      }).then(({ data }) => {
        console.log("成功获取到菜单数据:", data.data);
        this.menus = data.data;
      });
    },
    success(msg) {
      this.$message({
        message: msg,
        type: "success",
      });
    },
    error(msg) {
      this.$message({
        message: msg,
        type: "warning",
      });
    },
  },
  //生命周期 - 创建完成(可以访问当前this实例)
  created() {
    this.getMenus();
    Object.assign(this.copyCategory, this.category);
  },
  //生命周期 - 挂载完成(可以访问DOM元素)
  mounted() {},
  beforeCreate() {}, //生命周期 - 创建之前
  beforeMount() {}, //生命周期 - 挂载之前
  beforeUpdate() {}, //生命周期 - 更新之前
  updated() {}, //生命周期 - 更新之后
  beforeDestroy() {}, //生命周期 - 销毁之前
  destroyed() {}, //生命周期 - 销毁完成
  activated() {}, //如果页面有keep-alive缓存功能,这个函数会触发
};
</script>
<style  scoped>
</style>

4.批量删除

封装复用方法 闭包

<!--  -->
<template>
  <div>
    <el-switch
      v-model="isDraggable"
      active-text="开启拖拽"
      inactive-text="关闭拖拽"
    >
    </el-switch>
    <el-button type="danger" round @click="batchDelete">批量删除</el-button>
    <el-tree
      :data="menus"
      :props="defaultProps"
      show-checkbox
      node-key="catId"
      :expand-on-click-node="false"
      :default-expanded-keys="expandedkey"
      @node-expand="expend"
      @node-collapse="nodeClose"
      :draggable="isDraggable"
      :allow-drop="allowDrop"
      @node-drop="handleDrop"
      ref="menuTree"
    >
      <span class="custom-tree-node" slot-scope="{ node, data }">
        <!-- 插槽,代替了原来的 label  里所显示的内容  将插槽内容显示在原来的每个结点上面  -->
        <span>{{ node.label }}</span>
        <span>
          <el-button
            v-if="node.level <= 2"
            type="text"
            size="mini"
            @click="() => append(data)"
          >
            Append
          </el-button>
          <el-button type="text" size="mini" @click="edit(data)">
            Edit
          </el-button>

          <el-button
            v-if="node.childNodes.length == 0"
            type="text"
            size="mini"
            @click="() => remove(node, data)"
          >
            Delete
          </el-button>
        </span>
      </span>
    </el-tree>

    <el-dialog
      :title="submitTitle"
      :visible.sync="dialogVisible"
      :close-on-click-modal="false"
      width="30%"
    >
      <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>
      <div slot="footer" class="dialog-footer">
        <el-button @click="cancelForm">取 消</el-button>
        <el-button type="primary" @click="submitForm">确 定</el-button>
      </div>
    </el-dialog>
  </div>
</template>

<script>
//这里可以导入其他文件(比如:组件,工具js,第三方插件js,json文件,图片文件等等)
//例如:import 《组件名称》 from '《组件路径》';
class UpdateCategoryData {
  constructor(catId, parentCid, sort, catLevel) {
    this.catId = catId;
    this.parentCid = parentCid;
    this.sort = sort;
    this.catLevel = catLevel;
  }
}
export default {
  //import引入的组件需要注入到对象中才能使用
  components: {},
  props: {},
  data() {
    return {
      isDraggable: false,
      submitType: "",
      submitTitle: "",
      dialogVisible: false,
      menus: [],
      defaultProps: {
        children: "children",
        label: "name",
      },
      expandedkey: [],
      copyCategory: {},
      category: {
        catId: null,
        name: "",
        parentCid: 0,
        catLevel: 0,
        showStatus: 1,
        productUnit: "",
        icon: "",
        sort: 0,
      },
    };
  },
  //监听属性 类似于data概念
  computed: {},
  //监控data中的数据变化
  watch: {},
  //方法集合
  methods: {
    confirm(msg,success,error){
      this.$confirm(msg, "提示", {
        confirmButtonText: "确定",
        cancelButtonText: "取消",
        type: "warning",
      }).then(() => {
        success();
      }).catch(()=>{
        error();
      });
    },
    batchDelete() {
      let deleteArray = this.$refs.menuTree.getCheckedNodes();
      console.log("被选中的元素", deleteArray);
      let ids = deleteArray.map((c) => {
        return c.catId;
      });
      console.log("被选中的Id", ids);

      // this.$confirm(`是否批量删除`, "提示", {
      //   confirmButtonText: "确定",
      //   cancelButtonText: "取消",
      //   type: "warning",
      // }).then(() => {
        // this.$http({
        //   url: this.$http.adornUrl("/product/category/delete"),
        //   method: "post",
        //   data: this.$http.adornData(ids, false),
        // }).then(({ data }) => {
        //   this.success("批量删除成功");
        //   this.getMenus();
        // });
      // }).catch((error)=>{
      //   this.error("取消批量删除");
      // });

        this.confirm("是否批量删除",()=>{
          this.$http({
          url: this.$http.adornUrl("/product/category/delete"),
          method: "post",
          data: this.$http.adornData(ids, false),
        }).then(({ data }) => {
          this.success("批量删除成功");
          this.getMenus();
        });
        },()=>{
            this.error("取消批量删除");
        })


    },
    /**
     *
     * @param {*} draggingNode 拖拽的节点
     * @param {*} dropNode  拖拽到的哪个节点
     * @param {*} dropType 拖拽类型, 前后或者 内部
     * @param {*} event  事件对象
     */
    handleDrop(draggingNode, dropNode, dropType, event) {
      console.log(draggingNode, dropNode, dropType);

      //1.当前节点最新的父节点Id,
      let updateArray = new Array();

      let draggingNodeLevel = draggingNode.level;
      if (dropType == "inner") {
        let dropNodeId = dropNode.data.catId;
        let dropNodeLevel = dropNode.level;
        let childrenNew = dropNode.data.children;
        for (const index in childrenNew) {
          let { catId } = childrenNew[index];
          let updateCategoryData = new UpdateCategoryData(
            catId,
            dropNodeId,
            index,
            dropNodeLevel + 1
          );
          updateArray.push(updateCategoryData);
        }

        let div = dropNodeLevel + 1 - draggingNodeLevel;
        //递归把子节点都遍历完
        if (div != 0) {
          this.recursivelyTraverseChildNodes(
            draggingNode.data.children,
            div,
            updateArray
          );
        }
      } else {
        //往前插入节点或者后插入节点
        let parentLevel = dropNode.parent.level;
        console.log(parentLevel);
        let parentChildrenArr = {};
        if (parentLevel == 0) {
          parentChildrenArr = dropNode.parent.data;
        } else {
          parentChildrenArr = dropNode.parent.data.children;
        }

        let parentCid = dropNode.data.parentCid;
        for (const index in parentChildrenArr) {
          let { catId } = parentChildrenArr[index];
          let updateCategoryData = new UpdateCategoryData(
            catId,
            parentCid,
            index,
            parentLevel + 1
          );
          updateArray.push(updateCategoryData);
        }

        let div = parentLevel + 1 - draggingNodeLevel;
        console.log("parentLevel", parentLevel);
        console.log("draggingNodeLevel", draggingNodeLevel);
        //递归把子节点都遍历完
        if (div != 0) {
          this.recursivelyTraverseChildNodes(
            draggingNode.data.children,
            div,
            updateArray
          );
        }
      }

      console.log(updateArray);
      //发送http请求修改
      this.$http({
        url: this.$http.adornUrl("/product/category/updateList"),
        method: "post",
        data: this.$http.adornData(updateArray, false),
      }).then(({ data }) => {
        this.success("修改位置与排序成功");
        this.getMenus();
      });
    },
    recursivelyTraverseChildNodes(children, div, arr) {
      if (children == null || children.length == 0) {
        //没有子节点了
        return;
      } else {
        for (const child of children) {
          let updateCategoryData = new UpdateCategoryData(
            child.catId,
            child.parentCid,
            child.sort,
            child.catLevel + div
          );
          arr.push(updateCategoryData);
          this.recursivelyTraverseChildNodes(child.children, div, arr);
        }
      }
    },
    allowDrop(draggingNode, dropNode, type) {
      //1、被拖动的当前节点以及所在的父节点总层数不能大于3
      console.log("allowDrop", draggingNode, dropNode, type);
      let level = this.countNodeLevel(
        draggingNode.data,
        draggingNode.data.catLevel
      );
      if (type == "inner") {
        if (dropNode.level + level <= 3) {
          return true;
        } else {
          return false;
        }
      }

      if (type == "next") {
        if (dropNode.parent.level + level <= 3) {
          return true;
        } else {
          return false;
        }
      }

      if (type == "prev") {
        if (dropNode.parent.level + level <= 3) {
          return true;
        } else {
          return false;
        }
      }
      //1)被拖动的当前节点总层数
      return false;
    },
    //递归查找该节点加上子类最深层级一共有几层;包含自己算
    countNodeLevel(node, l) {
      let children = node.children;
      let levelMax = 0;
      if (children.length == 0) {
        return node.catLevel - l + 1;
      } else {
        for (let child of children) {
          let level = this.countNodeLevel(child, l);
          if (level > levelMax) {
            levelMax = level;
          }
        }
      }
      return levelMax;
    },
    resetCategory() {
      Object.assign(this.category, this.copyCategory);
      //如果你希望在 console.log 输出的时候看到对象的当前状态,
      //你可以在赋值操作之前进行 console.log,或者使用对象解构等方法创建一个新的对象进行输出,以确保输出的是当前状态的副本而不是对象的引用。
      let categoryre = {};
      Object.assign(categoryre, this.category);
      console.log("执行了重置", categoryre);
    },
    submitForm() {
      if (this.submitType == "add") {
        this.addCategory();
      }
      if (this.submitType == "edit") {
        this.updateCategory();
      }
    },
    updateCategory() {
      this.$http({
        url: this.$http.adornUrl("/product/category/update"),
        method: "post",
        data: this.$http.adornData(this.category, false),
      }).then(({ data }) => {
        this.success("修改成功");
        this.getMenus();
        this.resetCategory();
        this.dialogVisible = false;
      });
    },
    edit(data) {
      this.submitType = "edit";
      this.submitTitle = "修改分类菜单";
      console.log("正在修改数据", data);
      this.dialogVisible = true;

      //发送http请求获取回显数据
      this.$http({
        url: this.$http.adornUrl(`/product/category/info/${data.catId}`),
        method: "get",
      }).then(({ data }) => {
        let categoryEdit = data.data;
        console.log("回显数据", categoryEdit);
        this.category.catId = categoryEdit.catId;
        this.category.name = categoryEdit.name;
        this.category.parentCid = categoryEdit.parentCid;
        this.category.catLevel = categoryEdit.catLevel;
        this.category.showStatus = categoryEdit.showStatus;
        this.category.productUnit = categoryEdit.productUnit;
        this.category.icon = categoryEdit.icon;
        this.category.sort = categoryEdit.sort;
        console.log("category被回显数据", this.category);
      });
    },
    cancelForm() {
      this.resetCategory();
      this.dialogVisible = false;
    },
    append(data) {
      this.resetCategory();
      console.log("append", this.category);
      this.category.parentCid = data.catId;
      this.category.catLevel = data.catLevel + 1;
      this.category.showStatus = 1;
      this.category.sort = 0;
      this.dialogVisible = true;
      this.submitType = "add";
      this.submitTitle = "添加分类菜单";
    },
    addCategory() {
      console.log("提交三级分类的数据", this.category);

      this.$http({
        url: this.$http.adornUrl("/product/category/save"),
        method: "post",
        data: this.$http.adornData(this.category, false),
      }).then(({ data }) => {
        this.success("添加分类成功");
        this.getMenus();
      });

      this.resetCategory();
      this.dialogVisible = false;
    },
    remove(node, data) {
      console.log("remove", node, data);
      let ids = [data.catId];
      // console.log(this); //vue
      this.$confirm(`是否删除【${data.name}】菜单`, "提示", {
        confirmButtonText: "确定",
        cancelButtonText: "取消",
        type: "warning",
      })
        .then(() => {
          // console.log(this); //vue
          this.$http({
            url: this.$http.adornUrl("/product/category/delete"),
            method: "post",
            data: this.$http.adornData(ids, false),
          }).then(({ data }) => {
            // console.log(this); //vue
            this.$message({
              message: "菜单删除成功",
              type: "success",
            });
            // node.visible = false;
            this.getMenus();
            //设置需要默认展开的菜单
          });
        })
        .catch(() => {
          this.$message({
            message: "取消了删除",
            type: "warning",
          });
        });
    },
    expend(data, node, _) {
      console.log("展开了", node.data.catId);
      this.expandedkey.push(node.data.catId);
    },
    nodeClose(data, node, _) {
      let id = node.data.catId;
      console.log("收起了", id);
      let index = this.expandedkey.indexOf(id);
      this.expandedkey.splice(index, 1);
    },
    getMenus() {
      this.$http({
        url: this.$http.adornUrl("/product/category/list/tree"),
        method: "get",
      }).then(({ data }) => {
        console.log("成功获取到菜单数据:", data.data);
        this.menus = data.data;
      });
    },
    success(msg) {
      this.$message({
        message: msg,
        type: "success",
      });
    },
    error(msg) {
      this.$message({
        message: msg,
        type: "warning",
      });
    },
  },
  //生命周期 - 创建完成(可以访问当前this实例)
  created() {
    this.getMenus();
    Object.assign(this.copyCategory, this.category);
  },
  //生命周期 - 挂载完成(可以访问DOM元素)
  mounted() {},
  beforeCreate() {}, //生命周期 - 创建之前
  beforeMount() {}, //生命周期 - 挂载之前
  beforeUpdate() {}, //生命周期 - 更新之前
  updated() {}, //生命周期 - 更新之后
  beforeDestroy() {}, //生命周期 - 销毁之前
  destroyed() {}, //生命周期 - 销毁完成
  activated() {}, //如果页面有keep-alive缓存功能,这个函数会触发
};
</script>
<style  scoped>
</style>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/508055.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

2004-2022年上市公司企业战略激进度数据(含原始数据+计算代码+计算结果)

2004-2022年上市公司企业战略激进度数据&#xff08;含原始数据计算代码计算结果&#xff09; 1、时间2004-2022年 2、来源&#xff1a;原始数据整理自csmar 3、指标&#xff1a; 证券代码、统计截止日期、员工人数、证券简称、报表类型、固定资产净额、无形资产净额、资产…

算法学习——LeetCode力扣动态规划篇6(121. 买卖股票的最佳时机、122. 买卖股票的最佳时机 II、123. 买卖股票的最佳时机 III)

算法学习——LeetCode力扣动态规划篇6 121. 买卖股票的最佳时机 121. 买卖股票的最佳时机 - 力扣&#xff08;LeetCode&#xff09; 描述 给定一个数组 prices &#xff0c;它的第 i 个元素 prices[i] 表示一支给定股票第 i 天的价格。 你只能选择 某一天 买入这只股票&…

Transformers -- 深入研究 - part 3

公众号:Halo咯咯,欢迎关注~ 前文回顾: Transformers -- 以通俗易懂的方式解释 - Part 1Transformers -- 未知英雄 - Part 2世界正在为人工智能和生成式人工智能而疯狂,特别是 2023 年的 ChatGPT 和大型语言模型。在我们讨论本系列后续部分的技术细节之前,让我们先从它的想…

用Servlet实现一个简单的表白墙

1. 准备工作 创建项目,引入依赖...... 将静态页面放到项目中(放在webapp目录下): 当前,这个表白墙页面,已经可以输入内容,点击提交之后也能显示内容,后续后端要做的工作即: ①存档 用户点提交的时候,把刚才输入的内容通过网络传输给服务器,由服务器保存这个数据. ②读档 …

NAT地址转换内外网通信

实验要求&#xff1a;内网地址通过nat转换成外网地址&#xff0c;联通外网服务器&#xff0c;达到内网外网互通 拓扑结构&#xff1a; 配置完成后&#xff0c;在ar1的G1口设置抓包&#xff0c;在pc1设备上ping ar2的地址&#xff0c;通过查看抓包信息&#xff0c;可以看到访问…

49 el-input 的 模型 视图 双向同步

前言 这里来看一下 el-input 这边的 数据 和 视图的双向绑定 最开始 我以为 这部分的处理应该是 vue 这边实现的, 但是跟踪调试了一下 发现这部分的处理是业务这边 自己实现的 这部分 还是有一些 值得记录的东西, 从这里 要去理解的而是 vue 这边从宏观的框架上面来说 帮我们…

python如何画奥运五环

绘制奥运五环主要涉及到Python中的turtle绘图库运用&#xff1a; 程序源代码为&#xff1a; import turtle turtle.width(10) turtle.color(black) turtle.circle(50) turtle.penup() turtle.goto(120,0) turtle.pendown() turtle.color(red) turtle.circle(50) turtle.penup()…

HWOD:整型数组排序

一、知识点 while(1){}表示永久循环 使用break结束循环 二、题目 1、描述 输入整型数组和排序标识&#xff0c;对其元素按照升序或降序进行排序 2、数据范围 1<n<1000 0<val<100000 3、输入 第一行输入数组元素个数 第二行输入待排序的数组&#x…

第十四届蓝桥杯(八题C++ 题目+代码+注解)

目录 题目一&#xff08;日期统计 纯暴力&#xff09;&#xff1a; 代码&#xff1a; 题目二&#xff08;01串的熵 模拟&#xff09;&#xff1a; 代码&#xff1a; 题目三&#xff08;治炼金属&#xff09;&#xff1a; 代码&#xff1a; 题目四&#xff08;飞机降落 深度…

【JAVASE】学习数组的定义与使用

✅作者简介&#xff1a;大家好&#xff0c;我是橘橙黄又青&#xff0c;一个想要与大家共同进步的男人&#x1f609;&#x1f609; &#x1f34e;个人主页&#xff1a; 再无B&#xff5e;U&#xff5e;G-CSDN博客 目标&#xff1a; 1. 理解数组基本概念 2. 掌握数组的基本用法…

星际门计划:微软与OpenAI联手打造未来AI超级计算机

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

windows安装mysql

win r cmd 以管理员权限运行。数据库配置文件my.ini放到安装包里&#xff0c;配置文件内路径根据实际情况修改&#xff0c;配置文件字符集根据实际需要修改。 1、cd c:\mysql\bin切换目录 2、mysqld --initialize --console 初始化数据库&#xff0c;初始化完成…

[操作系统课设]GeeKOS操作系统的研究与实现

一.GeekOS操作系统概论 1.1教学操作系统 &#xff08;1&#xff09;针对RISC结构MIPS处理器 操作系统&#xff1a;Nachos、OS/161 &#xff08;2&#xff09;针对CISC结构Intel IA-32 (or x86)通用处理 操作系统&#xff1a;MINIX、GeekOS 我们用到的是&#xff1a;GeekOS 1&…

SpringMvc执行流程源码解析

一、简介 Spring web Mvc是基于ServletApi构建的原始Web模块&#xff0c;从一开始就包含在Spring框架中&#xff1b; 从Servlet到SpringMvc 最典型的MVc就是JSPServletjavaBean的模式&#xff1b; 弊端&#xff1a; 1、xml下配置Servlet的映射非常麻烦&#xff0c;效率低&…

OpenHarmony实战:命令行工具hdc安装应用指南

一、工具概述 hdc&#xff08;OpenHarmony Device Connector&#xff09;是为开发人员提供的用于设备连接调试的命令行工具&#xff0c;该工具需支持部署在 Windows/Linux/Mac 等系统上与 OpenHarmony 设备&#xff08;或模拟器&#xff09;进行连接调试通信。 简言之&#xf…

16进制的字符串转byte[]数组 以及将字节数组转换成十六进制的字符串

16进制的字符串转byte[]数组 public class ClientString16 {@Testpublic void get16Str(){String str="48 47 12 00 14 12 16 08 15 0d 30 0f 02 30 30 30 30 30 30 30 30 30 30 00 c2";byte[] bytes = hexStringToByteArray(str);getBytetoString(bytes);//String …

Redis实战篇-添加优惠卷

3.3 添加优惠卷 每个店铺都可以发布优惠券&#xff0c;分为平价券和特价券。平价券可以任意购买&#xff0c;而特价券需要秒杀抢购&#xff1a; tb_voucher&#xff1a;优惠券的基本信息&#xff0c;优惠金额、使用规则等 tb_seckill_voucher&#xff1a;优惠券的库存、开始抢…

(一)基于IDEA的JAVA基础10

相信最近许多朋友学习语言可能会有焦虑&#xff0c;“现在人工智能这么发达&#xff0c;丢个指令进去它就还给你一个结果&#xff0c;我们学习它还有意义吗&#xff1f;”。 对于这个问题&#xff0c;就像我们小学学习算数&#xff0c;我们明知道有计算器这么方便的东西&#…

Java 处理Mysql获取树形的数据

Mysql数据&#xff1a; 代码如下&#xff1a; Entity&#xff1a; Data Accessors(chain true) public class Region {private BigInteger id;//名称private String name;//父idprivate BigInteger parentId;private List<Region> children;private Integer createTim…

Spring 事件广播机制详解

前言 写这篇文章的初衷源自对 Spring 框架中事件机制的好奇心。在编写观察者模式代码示例时&#xff0c;我突然想起了 Spring 框架中支持多事件监听的特性&#xff0c;例如ContextClosedEvent、ContextRefreshedEvent、ContextStartedEvent等等。传统的观察者模式通常是基于单…