I have the following Vuex store (main.js):
import Vue from 'vue' import Vuex from 'vuex' Vue.use(Vuex) //init store const store = new Vuex.Store({ state: { globalError: '', user: { authenticated: false } }, mutations: { setGlobalError (state, error) { state.globalError = error } } }) //init app const app = new Vue({ router: Router, store, template: '<app></app>', components: { App } }).$mount('#app')
I also have the following routes defined for Vue Router (routes.js):
import Vue from 'vue' import VueRouter from 'vue-router' Vue.use(VueRouter) //define routes const routes = [ { path: '/home', name: 'Home', component: Home }, { path: '/login', name: 'Login', component: Login }, { path: '/secret', name: 'Secret', component: SecretPage, meta: { requiresLogin: true } ]
I'm trying to make it so that, if Vuex stores the user
object, and it has the authenticated
property set to false
, is has the router redirect the user to the login page.
I have this:
Router.beforeEach((to, from, next) => { if (to.matched.some(record => record.meta.requiresLogin) && ???) { // set Vuex state's globalError, then redirect next("/Login") } else { next() } })
The problem is I don't know how to access the Vuex store's user
object from inside the beforeEach
function.
I know that I can have the router guard logic inside components using BeforeRouteEnter
, but that would clutter up each component. I want to define it centrally at the router level instead.
To access Vuex state when defining Vue Router routes, we can import the store in the file with the Vue Router hooks. import Vuex from "vuex"; const store = new Vuex.
At the center of every Vuex application is the store. A "store" is basically a container that holds your application state.
vuex-router-sync is a helper to sync the router state to the centralized state store. Now all the views can be built from the state store and we don't need to check this. $router .
As suggested here, what you can do is to export your store from the file it is in and import it in the routes.js. It will be something like following:
You have one store.js:
import Vuex from 'vuex' //init store const store = new Vuex.Store({ state: { globalError: '', user: { authenticated: false } }, mutations: { setGlobalError (state, error) { state.globalError = error } } }) export default store
Now in routes.js, you can have:
import Vue from 'vue' import VueRouter from 'vue-router' import store from ./store.js Vue.use(VueRouter) //define routes const routes = [ { path: '/home', name: 'Home', component: Home }, { path: '/login', name: 'Login', component: Login }, { path: '/secret', name: 'Secret', component: SecretPage, meta: { requiresLogin: true } ] Router.beforeEach((to, from, next) => { if (to.matched.some(record => record.meta.requiresLogin) && ???) { // You can use store variable here to access globalError or commit mutation next("/Login") } else { next() } })
In main.js also you can import store
:
import Vue from 'vue' import Vuex from 'vuex' Vue.use(Vuex) import store from './store.js' //init app const app = new Vue({ router: Router, store, template: '<app></app>', components: { App } }).$mount('#app')
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With