Vue изменений не обнаружено в массиве - PullRequest
0 голосов
/ 27 февраля 2020

Привет, ребята, я работаю с 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>

Ответы [ 2 ]

1 голос
/ 27 февраля 2020

причина, по которой вы не видите обновления, состоит в том, что у массива нет причин для пересчета. Вы обновляете this.$set(item, "expand", false); объект, который не является реактивным. Причина, по которой он не реагирует, заключается в том, что вы не используете метод $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
    }

, который вы можете увидеть в действии здесь


В качестве альтернативы, .. .

Вот код, который может работать для вас, который не зависит от реактивности

Обратите внимание, что:

  • Я пересчитываю и переназначаю массив с 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>

увидеть это в действии здесь

1 голос
/ 27 февраля 2020

Если item объект не имеет начального свойства expand, вы должны объявить его наблюдаемым, используя this.$set(item, ...).

Прямое добавление нового свойства, например item.expand = ..., пропускает этот шаг и this.$set игнорирует это. Такое свойство не будет реактивным.

Подробнее об этом здесь: https://vuejs.org/v2/guide/reactivity.html

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...