javascript - 数组上未检测到 Vue 更改

标签 javascript vue.js vuejs2

大家好,我正在使用 Vue,我正在尝试从平面列表中创建递归树,我想在有人点击每个项目时切换其扩展属性,但由于某种原因它没有改变

它发生在这个函数中

  expandNode(item) {
      console.log("HERERERER");
      item.expand = false;
      this.$set(item, "expand", false);
    }

我希望我的数组具有反应性,但由于某种原因它不是,这可能是我重新打包数据或其他东西的方式,有人可以看一下吗??

这是我的 CodeSandbox

https://codesandbox.io/s/condescending-tree-51rbs

这是组件代码

<template>
  <div class="hello">
    <h1>{{ msg }}</h1>
    <tr v-for="(item ,index)  in flatArray" :key="index">
      <div class="item" @click="expandNode(item)">
        <div class="element" v-show="item.expand">
          {{ item.expand }}
          <span>{{ item.label }}</span>
        </div>
      </div>
    </tr>
  </div>
</template>

<script>
export default {
  name: "HelloWorld",
  props: {
    msg: String,
    data: { default: () => null, type: Array }
  },
  data() {
    return {
      flatArray: []
    };
  },
  mounted() {
    let arr = [];
    console.log("HERERER");
    this.recursive(this.data, arr, 0, null, -1);
    this.flatArray = arr;
    console.log(this.flatArray);
  },
  computed: {
    setPadding(item) {
      return `padding-left: ${item.level * 30}px;`;
    }
  },
  methods: {
    recursive(obj, newObj, level, parent, parentIndex) {
      obj.forEach(node => {
        if (node.children && node.children.length != 0) {
          node.level = level;
          node.leaf = false;
          node.expand = true;
          node.parent = parent;
          node.parentIndex = parent ? parentIndex : null;
          newObj.push(node);
          this.recursive(
            node.children,
            newObj,
            node.level + 1,
            node,
            newObj.indexOf(node)
          );
        } else {
          node.level = level;
          node.leaf = true;
          node.expand = true;
          node.parent = obj;
          node.parentIndex = parent ? parentIndex : null;
          newObj.push(node);
          return false;
        }
      });
    },
    expandNode(item) {
      console.log("HERERERER");
      item.expand = false;
      this.$set(item, "expand", false);
    }
  }
};
</script>

最佳答案

您没有看到更新的原因是数组没有理由重新计算。您正在更新 this.$set(item, "expand", false); 一个非响应式对象。它不是 react 性的原因是因为您在创建对象时没有使用 $set 方法。

如果在对象创建过程中正确使用 $set,结果会是这样。

    recursive(obj, newObj, level, parent, parentIndex) {
      obj.forEach(node => {
        this.$set(node, "level", level);
        this.$set(node, "expand", true);
        this.$set(node, "parentIndex", parent ? parentIndex : null);
        if (node.children && node.children.length !== 0) {
          this.$set(node, "leaf", false);
          this.$set(node, "parent", parent);
          newObj.push(node);
          this.recursive(
            node.children,
            newObj,
            node.level + 1,
            node,
            newObj.indexOf(node)
          );
        } else {
          this.$set(node, "leaf", true);
          this.$set(node, "parent", obj);
          newObj.push(node);
          return false;
        }
      });
    },

请注意,您现在可以使用item.expand = false

    expandNode(item) {
      item.expand = false;
      // this.$set(item, "expand", false);  <== not needed
    }

你可以在行动中看到here

<小时/>

或者,...

以下代码可能适合您,但不依赖于 react 性

请注意:

  • 我正在使用 this.flatArray = flattenTree(this.data); 重新计算并重新分配数组
  • 这个想法是,嵌套对象是“事实来源”,并且展平数组允许模板渲染。
<template>
  <div class="hello">
    <tr v-for="(item ,index) in flatArray" :key="index">
      <div
        @click="toggleExpandNode(item)"
        class="item"
        :style="{'margin-left':item.level * 1.6 +'em'}"
      >
        <div class="element">
          <span v-if="item.leaf">&#9900;</span>
          <span v-else-if="item.expand">&#9662;</span>
          <span v-else>&#9656;</span>
          &nbsp;
          <span>{{ item.label }}</span>
        </div>
      </div>
    </tr>
  </div>
</template>

<script>
const flattenTree = obj => {
  const flatTreeArr = [];
  let depth = 0;

  const flatten = (node, parentNode) => {
    flatTreeArr.push(node);
    node.level = depth;
    node.leaf = true;
    node.parent = parentNode;
    node.expand = node.expand === undefined ? true : node.expand;
    if (node.children) {
      node.leaf = false;
      if (node.expand) {
        depth++;
        node.children.forEach(br => flatten(br, node));
        depth--;
      }
    }
  };

  obj.forEach(br => flatten(br, null));
  return flatTreeArr;
};

export default {
  props: {
    data: { default: () => null, type: Array }
  },
  data() {
    return {
      flatArray: []
    };
  },
  mounted() {
    this.flatArray = flattenTree(this.data);
  },
  methods: {
    toggleExpandNode(item) {
      item.expand = !item.expand;
      this.flatArray = flattenTree(this.data);
    }
  }
};
</script>

查看实际效果 here

关于javascript - 数组上未检测到 Vue 更改,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60436063/

相关文章:

javascript - 如果用户已登录,则更改路由名称和组件

javascript - 测试VueJS组件时如何将数据设置到this.$root?

vue.js - 如何在应用程序外部安装路由器链接?

javascript - 让Vue注册datepicker事件

javascript - Vuejs 自动将所有 Prop 传递给 child

javascript - 如何为搜索输入制作 select2 占位符

javascript - 删除除 0-9 + -/* 和 ^ 之外的任何字符

javascript - 使用 jsdoc 记录匿名对象和函数的最佳方式

javascript - 如何在不丢失链接属性的情况下将超文本链接复制到剪贴板

vue.js - 如何禁用 Vue 转换组等待浏览器选项卡被选择?