最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Vuex getters in computed properties showing undefined until full page load - Stack Overflow

programmeradmin1浏览0评论

I'm trying to create puted properties using data pulled with the mapGetters function in vuex, but I'm always getting undefined, until the page/dom fully loads.

Here's an example for an isRegistered puted property that I use to hide/display certain buttons.

puted: {
      ...mapGetters(['solos', 'user']),
      isRegistered () {
        return this.solos.registered.indexOf(this.user._id) !== -1
      }
}

Here is the HTML for buttons that use the isRegistered puted property.

<a href="#" class="register-button" v-if="!isRegistered">REGISTER NOW</a>
<a href="#" class="registered-button" v-if="isRegistered">REGISTERED</a>

I'm setting the getters via an action which I'm calling in the created function

created () {
      this.getTournament('solos').catch(err => {})
}

Here's my action for setting the corresponding getters

getTournament: ({mit}, type) => {
    return feathers.service('tournaments').find({
      query: {
        type: type,
        status: 'registering'
      }
    }).then(tourney => {
      mit(type.toUpperCase(), tourney.data[0])
    })
}

Here's the corresponding mutation

const mutations = {
  SOLOS (state, result) {
    state.solos = result;
  }
};

Here's the corresponding getter

const getters = {
   solos (state) {
     return state.solos
   }
}

The issue I'm having is the value is initially showed as undefined until the PAGE/DOM is fully loaded, causing .indexOf to throw the following error:

TypeError: Cannot read property 'indexOf' of undefined

The only way I've been able to get this to work is to use an if statement on the puted property that checks if the state data has loaded yet.

isRegistered () {
  if (this.solos._id) return this.solos.registered.indexOf(this.user._id) !== -1
}

This doesn't seem like a proper way of doing things. What am I doing wrong?

I'm trying to create puted properties using data pulled with the mapGetters function in vuex, but I'm always getting undefined, until the page/dom fully loads.

Here's an example for an isRegistered puted property that I use to hide/display certain buttons.

puted: {
      ...mapGetters(['solos', 'user']),
      isRegistered () {
        return this.solos.registered.indexOf(this.user._id) !== -1
      }
}

Here is the HTML for buttons that use the isRegistered puted property.

<a href="#" class="register-button" v-if="!isRegistered">REGISTER NOW</a>
<a href="#" class="registered-button" v-if="isRegistered">REGISTERED</a>

I'm setting the getters via an action which I'm calling in the created function

created () {
      this.getTournament('solos').catch(err => {})
}

Here's my action for setting the corresponding getters

getTournament: ({mit}, type) => {
    return feathers.service('tournaments').find({
      query: {
        type: type,
        status: 'registering'
      }
    }).then(tourney => {
      mit(type.toUpperCase(), tourney.data[0])
    })
}

Here's the corresponding mutation

const mutations = {
  SOLOS (state, result) {
    state.solos = result;
  }
};

Here's the corresponding getter

const getters = {
   solos (state) {
     return state.solos
   }
}

The issue I'm having is the value is initially showed as undefined until the PAGE/DOM is fully loaded, causing .indexOf to throw the following error:

TypeError: Cannot read property 'indexOf' of undefined

The only way I've been able to get this to work is to use an if statement on the puted property that checks if the state data has loaded yet.

isRegistered () {
  if (this.solos._id) return this.solos.registered.indexOf(this.user._id) !== -1
}

This doesn't seem like a proper way of doing things. What am I doing wrong?

Share Improve this question edited May 25, 2017 at 17:40 Jayson H asked May 25, 2017 at 17:31 Jayson HJayson H 2,0485 gold badges22 silver badges30 bronze badges 3
  • 1 This is the proper way of doing things. Javascript is defensive, you have to check if objects are in place before accessing inner object properties. – Egor Stambakio Commented May 25, 2017 at 17:49
  • 1 Your last bit of code is pretty much exactly how you should do it. – Bert Commented May 25, 2017 at 17:49
  • Okay, thanks to the both of you! – Jayson H Commented May 25, 2017 at 17:52
Add a ment  | 

1 Answer 1

Reset to default 12

The issue is probably about the vue lifecycle.

At the created hook, the puted properties, watchers and data of ponent are initialized (runs synchronously), but solos getter runs inside a promise (runs asynchronously), so the lifecycle continue without the promise get done.

When created hook is finished, the watchers was created, which means if data get changed, they will be reflected at the DOM, but only after the promise returns the result, the solos will be filled. So at the initialization of the app, the promise is not ended, but as soon it ends the data is updated and displayed.

A way to avoid this error is to use v-if="solos.registered", but I'm still searching for a better way to solve this.

发布评论

评论列表(0)

  1. 暂无评论