Drücke „Enter”, um zum Inhalt zu springen.

VUE.JS: Array gefüllt mit Ergebnissen aus Asynchronen Abrufen wird nicht verwendet

Manuel 0

Mehrere Stunden hat mich ein Problem gekostet, dessen Lösung am Ende dann recht einfach war.

Das Problem

Ich habe ein Component, in welchem die Daten

  data: function() {
      return {
          peopleFile: {},
      };
  },

definiert sind und über die Methode getPeople Daten abgerufen werden, die in Schleife dann zu weiteren Datenabrufen verwendet werden:

getPeople: async function() {
    this.$root.loading = true;
    await people..where('searchglobal', this.entrysSearch).page('size', this.entrysPerPage).page('number', this.page).orderBy(this.entrysOrderBy).get().then(response => {
        this.people = response;
        for (var i = 0; i < this.contacts.data.length; i++) { this.getPeopleFile(this.people.data[i]); } this.$root.loading = false; }).catch(error => {
        this.$root.loading = false;
    });
}, getPeopleFile: async function(peopleObject) {
    peopleObject.PeopleFile().first().then(response => {
        this.peopleFile[peopleObject.id] = response.image.encoded;
    }).catch(error => {
        this.peopleFile[peopleObject.id] = false;
    });
},

Für den Abruf verwendet ich die Bibliothek vue-api-query (https://github.com/robsontenorio/vue-api-query) für welche ich bereits die passenden und auch funktionierenden Models definiert habe.
Die Rückmeldung von getPeople sieht in etwa so aus

{
    "current_page": 1,
    "data": [{
        "id": 2,
        "name": "IT",
    }, ],
    "first_page_url": "http://127.0.0.1/people?sort=name&page%5Bsize%5D=100&page%5Bnumber%5D=1",
    "from": 1,
    "last_page": 1,
    "last_page_url": "http://127.0.0.1/people?sort=name&page%5Bsize%5D=100&page%5Bnumber%5D=1",
    "next_page_url": null,
    "path": "http://127.0.0.1/people",
    "per_page": 100,
    "prev_page_url": null,
    "to": 1,
    "total": 1
}

während die Rückmeldung von getPeopleFile so aussieht:

{
  "current_page": 1,
  "data": [
    {
      "id": 4,
      "name": "test",
	  "parentId": "2",
      "image": {
        "encoded": "",
        "mime": "image/jpeg",
      }
    }
  ],
  "first_page_url": "http://127.0.0.1/peopleFile/2/files?page%5Bnumber%5D=1",
  "from": 1,
  "last_page": 1,
  "last_page_url": "http://127.0.0.1/peopleFile/2/files?page%5Bnumber%5D=1",
  "next_page_url": null,
  "path": "http://127.0.0.1/production/peopleFile/2/files",
  "per_page": 25,
  "prev_page_url": null,
  "to": 1,
  "total": 1
}

Es gibt in der Component ein View, das wie folgt definiert ist und auf die Daten von peopleFile zurückgreifen soll:


<div class="col-lg-3 col-md-3" v-for="(contact, contactNo) in contacts">
  
<div class="thumbnail">
    
<div class="box-responsible">
      <img class="img-responsive" :src="peopleFile[contact.id]"> {{ contact.id }}
    </div>

  </div>

</div>

Problem an der Stelle nun: Er wollte einfach nicht auf peopleFile zugreifen, egal was ich gemacht habe.

Die Lösung

Ursächlich für das Problem ist wohl an der Stelle, dass mehrere ansynchrone Aufrufe stattfinden können, die dann dafür sorgen dass VUE.JS nicht mehr reactive auf die Variable peopleFile reagiert. Hierfür gibt es aber auch eine recht einfache Lösung:

          Axios.all(
              this.contacts.data.map(item =>
                  item
                  .PeopleFile()
                  .first()
              )
          ).then(
              Axios.spread((...PeopleFileLists) => {
                  this.peopleFile = PeopleFileLists;
              })
          );

Bei diesem Aufruf werden nun die einzelnen Objekte, abgerufen aus getPeople an Axios übergeben welches dann wiederum die mehreren Aufrufe zusammenfasst und das Ergebnis in PeopleFileList speichert, was man in peopleFile ablegen kann und sich dann reactive verhält.

Schreibe einen Kommentar

Deine E-Mail-Adresse wird nicht veröffentlicht. Erforderliche Felder sind mit * markiert.

Diese Website verwendet Akismet, um Spam zu reduzieren. Erfahre mehr darüber, wie deine Kommentardaten verarbeitet werden.