javascript - vue.js 子组件没有更新
问题描述
我有一个父组件和一个子组件。父组件“邀请”包含多个“邀请”子组件。当我刷新网站时,所有邀请都会显示在网站上,但是一旦我更新了邀请数组(添加邀请),它就不会显示在页面上,即使我可以在 vue 开发人员上看到正在更新的数组安慰。我必须刷新页面才能看到它。
当某些东西被添加到数组(更新)时,它看起来像这样: 见这里
当我将页面添加到数组后刷新页面时,它看起来像这样。
这是我的两个组件:
邀请.vue
<template>
<li class="nav-item dropdown" id="invitations">
<a class="nav-link dropdown-toggle" href="#" id="navbarDropdown" role="button"
data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
Invitations <i class="far fa-bell"></i>
</a>
<div class="dropdown-menu" style="min-width: 20rem" aria-labelledby="navbarDropdown"
id="dropdownMenu">
<div v-for="invitation in invitations" v-bind:key="invitation.id" v-
bind:invitations="invitations">
<Invitation :invitation="invitation" v-on:accept-invitation="accept" v-on:reject-
invitation="reject" v-bind:invitation="invitation"></Invitation>
</div>
</div>
</li>
</template>
<script>
import Invitation from "./Invitation";
export default {
name: "Invitations",
props: {
invitations: {
type: Array,
default: [],
required: true
},
user: {
type: Object,
required: true
}
},
methods: {
accept(team_id) {
axios.get('/request/accept/' + team_id)
.then((response) => {
this.invitations = this.invitations.filter(invitation => invitation.id)
});
},
reject(team_id) {
axios.get('/request/reject/' + team_id)
.then((response) => {
this.invitations = this.invitations.filter(invitation => invitation.id)
});
},
handleIncoming(invitation) {
this.saveNewInvitation(invitation);
return;
},
saveNewInvitation(invitation) {
this.invitations.push(invitation);
},
},
mounted() {
Echo.channel('invitations.' + this.user.id)
.listen('NewInvitation', (e) => {
this.handleIncoming(e);
});
},
components: {Invitation}
}
</script>
邀请.vue
<template>
<div :id="invitation.id" class="mb-2">
<a class="ml-4" title="Show team info" style="margin-right: 90px"
:href="'/team/index/' + invitation.id" v-bind:name="invitation.name">{{invitation.name}}</a>
<a class="float-right mr-3" href="#" @click="$emit('reject-invitation', invitation.id)">
<i class="fas fa-times reject" :teamId="invitation.id"></i>
</a>
<a class="float-right mr-3" href="#" @click="$emit('accept-invitation', invitation.id)">
<i class="fas fa-check accept" :teamId="invitation.id"></i>
</a>
</div>
</template>
<script>
export default {
name: "Invitation",
props: {
invitation: {
type: Object,
default: null
}
}
}
</script>
有人可以告诉我有什么问题吗?
解决方案
我不是 100% 你专门用来修改数组的哪个函数,但我注意到一个潜在的挂断的地方是......
saveNewInvitation(invitation) {
this.invitations.push(invitation);
}
在 Vue 文档中(链接到“Reactivity in Depth -- For Arrays”)Vue 特别警告它不会注意到底层数组的突变。
Vue 无法检测到数组的以下更改:
当您直接使用索引设置项目时,例如
vm.items[indexOfItem] = newValue
当您修改数组的长度时,例如
vm.items.length = newLength
虽然.push()
可能不会立即触发警报,但当您阅读 push 方法的文档(链接到 MDN 文档)时,您会注意到它正在改变数组,而不是创建新数组。
push() 方法将一个或多个元素添加到数组的末尾,并返回数组的新长度。[重点补充]
正如之前的 Stack Overflow 问题(讨论链接)中所见,push 方法的内部工作只做两件事——改变长度,并按索引设置项目——这是 Vue 唯一看不到的两件事.
将创建一个新的 Array 对象的一种方法是concat()
:
saveNewInvitation(invitation) {
this.invitations = this.invitations.concat(invitation)
}
...因为,如concat()
(链接到 MDN 文档)的文档中所述:
返回值:一个新的 Array 实例。
... Vue 会正确看到。