J'utilise une propriété calculée Vue.js mais je rencontre un problème: la méthode calculée EST appelée au bon moment, mais la valeur renvoyée par la méthode calculée est ignorée!
Ma méthode
computed: {
filteredClasses() {
let classes = this.project.classes
const ret = classes && classes.map(klass => {
const klassRet = Object.assign({}, klass)
klassRet.methods = klass.methods.filter(meth => this.isFiltered(meth, klass))
return klassRet
})
console.log(JSON.stringify(ret))
return ret
}
}
Les valeurs imprimées par le console.log
les instructions sont correctes, mais lorsque j'utilise filteredClasses
dans le modèle, il utilise simplement première valeur mise en cache et ne met jamais à jour le modèle. Ceci est confirmé par Vue chrome devtools (filteredClasses
ne change jamais après la mise en cache initiale).
Quelqu'un pourrait-il me donner des informations sur la raison pour laquelle cela se produit?
Project.vue
<template>
<div>
<div class="card light-blue white-text">
<div class="card-content row">
<div class="col s4 input-field-white inline">
<input type="text" v-model="filter.name" id="filter-name">
<label for="filter-name">Name</label>
</div>
<div class="col s2 input-field-white inline">
<input type="text" v-model="filter.status" id="filter-status">
<label for="filter-status">Status (PASS or FAIL)</label>
</div>
<div class="col s2 input-field-white inline">
<input type="text" v-model="filter.apkVersion" id="filter-apkVersion">
<label for="filter-apkVersion">APK Version</label>
</div>
<div class="col s4 input-field-white inline">
<input type="text" v-model="filter.executionStatus" id="filter-executionStatus">
<label for="filter-executionStatus">Execution Status (RUNNING, QUEUED, or IDLE)</label>
</div>
</div>
</div>
<div v-for="(klass, classIndex) in filteredClasses">
<ClassView :klass-raw="klass"/>
</div>
</div>
</template>
<script>
import ClassView from "./ClassView.vue"
export default {
name: "ProjectView",
props: {
projectId: {
type: String,
default() {
return this.$route.params.id
}
}
},
data() {
return {
project: {},
filter: {
name: "",
status: "",
apkVersion: "",
executionStatus: ""
}
}
},
async created() {
// Get initial data
const res = await this.$lokka.query(`{
project(id: "${this.projectId}") {
name
classes {
name
methods {
id
name
reports
executionStatus
}
}
}
}`)
// Augment this data with latestReport and expanded
const reportPromises = []
const reportMeta = []
for(let i = 0; i < res.project.classes.length; ++i) {
const klass = res.project.classes[i];
for(let j = 0; j < klass.methods.length; ++j) {
res.project.classes[i].methods[j].expanded = false
const meth = klass.methods[j]
if(meth.reports && meth.reports.length) {
reportPromises.Push(
this.$lokka.query(`{
report(id: "${meth.reports[meth.reports.length-1]}") {
id
status
apkVersion
steps {
status platform message time
}
}
}`)
.then(res => res.report)
)
reportMeta.Push({
classIndex: i,
methodIndex: j
})
}
}
}
// Send all report requests in parallel
const reports = await Promise.all(reportPromises)
for(let i = 0; i < reports.length; ++i) {
const {classIndex, methodIndex} = reportMeta[i]
res.project.classes[classIndex]
.methods[methodIndex]
.latestReport = reports[i]
}
this.project = res.project
// Establish WebSocket connection and set up event handlers
this.registerExecutorSocket()
},
computed: {
filteredClasses() {
let classes = this.project.classes
const ret = classes && classes.map(klass => {
const klassRet = Object.assign({}, klass)
klassRet.methods = klass.methods.filter(meth => this.isFiltered(meth, klass))
return klassRet
})
console.log(JSON.stringify(ret))
return ret
}
},
methods: {
isFiltered(method, klass) {
const nameFilter = this.testFilter(
this.filter.name,
klass.name + "." + method.name
)
const statusFilter = this.testFilter(
this.filter.status,
method.latestReport && method.latestReport.status
)
const apkVersionFilter = this.testFilter(
this.filter.apkVersion,
method.latestReport && method.latestReport.apkVersion
)
const executionStatusFilter = this.testFilter(
this.filter.executionStatus,
method.executionStatus
)
return nameFilter && statusFilter && apkVersionFilter && executionStatusFilter
},
testFilter(filter, item) {
item = item || ""
let outerRet = !filter ||
// Split on '&' operator
filter.toLowerCase().split("&").map(x => x.trim()).map(seg =>
// Split on '|' operator
seg.split("|").map(x => x.trim()).map(segment => {
let quoted = false, postOp = x => x
// Check for negation
if(segment.indexOf("!") === 0) {
if(segment.length > 1) {
segment = segment.slice(1, segment.length)
postOp = x => !x
}
}
// Check for quoted
if(segment.indexOf("'") === 0 || segment.indexOf("\"") === 0) {
if(segment[segment.length-1] === segment[0]) {
segment = segment.slice(1, segment.length-1)
quoted = true
}
}
if(!quoted || segment !== "") {
//console.log(`Item: ${item}, Segment: ${segment}`)
//console.log(`Result: ${item.toLowerCase().includes(segment)}`)
//console.log(`Result': ${postOp(item.toLowerCase().includes(segment))}`)
}
let innerRet = quoted && segment === "" ?
postOp(!item) :
postOp(item.toLowerCase().includes(segment))
//console.log(`InnerRet(${filter}, ${item}): ${innerRet}`)
return innerRet
}).reduce((x, y) => x || y, false)
).reduce((x, y) => x && y, true)
//console.log(`OuterRet(${filter}, ${item}): ${outerRet}`)
return outerRet
},
execute(methID, klassI, methI) {
this.project.classes[klassI].methods[methI].executionStatus = "QUEUED"
// Make HTTP request to execute method
this.$http.post("/api/Method/" + methID + "/Execute")
.then(response => {
}, error =>
console.log("Couldn't execute Test: " + JSON.stringify(error))
)
},
registerExecutorSocket() {
const socket = new WebSocket("ws://localhost:4567/api/Executor/")
socket.onmessage = msg => {
const {methodID, report, executionStatus} = JSON.parse(msg.data)
for(let i = 0; i < this.project.classes.length; ++i) {
const klass = this.project.classes[i]
for(let j = 0; j < klass.methods.length; ++j) {
const meth = klass.methods[j]
if(meth.id === methodID) {
if(report)
this.project.classes[i].methods[j].latestReport = report
if(executionStatus)
this.project.classes[i].methods[j].executionStatus = executionStatus
return
}
}
}
}
},
prettyName: function(name) {
const split = name.split(".")
return split[split.length-1]
}
},
components: {
"ClassView": ClassView
}
}
</script>
<style scoped>
</style>
J'ai rencontré un problème similaire auparavant et l'ai résolu en utilisant une méthode régulière au lieu d'une propriété calculée. Déplacez tout simplement dans une méthode et renvoyez votre ret. Documents officiels.
Vous devez attribuer une valeur de clé unique aux éléments de la liste dans le v-for. Ainsi..
<ClassView :klass-raw="klass" :key="klass.id"/>
Sinon, Vue ne sait pas quels éléments mettre à jour. Explication ici https://vuejs.org/v2/guide/list.html#key
Si votre intention est que la propriété calculée soit mise à jour lorsque project.classes.someSubProperty
change, cette sous-propriété doit exister lorsque la propriété calculée est définie. Vue ne peut pas détecter l'ajout ou la suppression de propriétés , uniquement les modifications apportées aux propriétés existantes.
Cela m'a mordu lors de l'utilisation d'un magasin Vuex avec un objet state
vide. Mes modifications ultérieures de l'état n'entraîneraient pas de propriétés calculées qui dépendent de sa réévaluation. L'ajout de clés explicites avec des valeurs nulles à l'état Veux a résolu ce problème.
Je ne sais pas si les clés explicites sont réalisables dans votre cas, mais cela pourrait aider à expliquer pourquoi la propriété calculée devient périmée.
Documents réactifs Vue, pour plus d'informations: https://vuejs.org/v2/guide/reactivity.html#Change-Detection-Caveats
Si vous ajoutez console.log avant de revenir, vous pourrez peut-être voir la valeur calculée dans filteredClasses
.
Mais DOM ne sera pas mis à jour pour une raison quelconque.
Ensuite, vous devez forcer le rendu de DOM.
La meilleure façon de restituer est juste en ajoutant une clé comme valeur calculée comme ci-dessous.
<div
:key="JSON.stringify(filteredClasses)"
v-for="(klass, classIndex) in filteredClasses"
>
<ClassView
:key="classIndex"
:klass-raw="klass"
/>
</div>
Attention:
N'utilisez pas de valeurs non primitives comme des objets et des tableaux comme clés. Utilisez plutôt des chaînes ou des valeurs numériques.
C'est pourquoi j'ai converti le tableau filteredClasses
en chaîne. (Il peut y avoir d'autres méthodes de conversion de tableau-> chaîne)
Et je veux aussi dire que "Il est recommandé de fournir un attribut clé avec v-for chaque fois que possible".