#
Quasar Boot Files | webally.co.za
#
Creating a new boot file
quasar new boot <name> [--format ts]
Where <name>
should be exchanged by a suitable name for your boot file.
This command creates a new file: /src/boot/<name>
.js with the following content:
// import something here
// "async" is optional!
// remove it if you don't need it
export default async ({ /* app, router, store */ }) => {
// something to do
}
#
Exports inherit an object with the following properties
A boot file is a simple JavaScript file which can optionally export a function. Quasar will then call the exported function when it boots the application and additionally pass an object
with the following properties to the function:
#
Scaffold for example
export default ({ app,router,store,ssrContext,urlPath,publicPath,redirect }) => {
// something to do
}
Boot files can also be async:
export default async ({ app, router, store }) => {
// something to do
await something()
}
You can wrap the returned function with boot helper to get a better IDE autocomplete experience (through Typescript):
import { boot } from 'quasar/wrappers'
export default boot(async ({ app, router, store }) => {
// something to do
await something()
})
Notice we are using the ES6 destructuring assignment. Only assign what you actually need/use.
You may ask yourself why we need to export a function. This is actually optional, but before you decide to remove the default export, you need to understand when you need it:
// Outside of default export:
// - Code here gets executed immediately,
// - Good place for import statements,
// - No access to router, Vuex store, ...
export default async ({ app, router, store }) => {
// Code here has access to the Object param above, connecting
// with other parts of your app;
// Code here can be async (use async/await or directly return a Promise);
// Code here gets executed by Quasar CLI at the correct time in app's lifecycle:
// - we have a Router instantiated,
// - we have the optional Vuex store instantiated,
// - we have the root app's component ["app" prop in Object param] Object with
// which Quasar will instantiate the Vue app
// ("new Vue(app)" -- do NOT call this by yourself),
// - ...
}
#
When to use boot files
WARNING Please make sure you understand what problem boot files solve and when it is appropriate to use them, to avoid applying them in cases where they are not needed.
Boot files fulfill one special purpose: they run code before
the App’s Vue root component is instantiated while giving you access to certain variables, which is required if you need to initialize a library, interfere with Vue Router, inject Vue prototype or inject the root instance of the Vue app.
#
Examples of appropriate usage of boot files
- Your Vue plugin has installation instructions, like needing to call
app.use()
on it. - Your Vue plugin requires instantiation of data that is added to the root instance - An example would be
vue-i18n
. - You want to add a global mixin using
app.mixin()
. - You want to add something to the Vue app globalProperties for convenient access - An example would be to conveniently use -
this.$axios
(for Options API) inside your Vue files instead of importing Axios in each such file. - You want to interfere with the router - An example would be to use
router.beforeEach
for authentication - You want to interfere with the Vuex store instance - An example would be to use
vuex-router-sync
package - Configure aspects of libraries - An example would be to create an instance of Axios with a base URL; you can then inject it into Vue prototype and/or export it (so you can import the instance from anywhere else in your app)
#
Example of unneeded usage of boot files
For plain JavaScript libraries like Lodash, which don’t need any initialization prior to their usage. Lodash, for example, might make sense to use as a boot file only if you want to inject Vue prototype with it, like being able to use this.$_ inside your Vue files.
#
Usage of boot files
The first step is always to generate a new boot file using Quasar CLI:
quasar new boot <name> [--format ts]
Where '<name>'
should be exchanged by a suitable name for your boot file.
This command creates a new file: /src/boot/<name>.js
with the following content:
// import something here
// "async" is optional!
// remove it if you don't need it
export default async ({ /* app, router, store */ }) => {
// something to do
}
You can also return a Promise:
// import something here
export default ({ /* app, router, store */ }) => {
return new Promise((resolve, reject) => {
// do something
})
}
TIP The default export can be left out of the boot file if you don’t need it. These are the cases where you don’t need to access the “app”, “router”, “store” and so on.
You can now add content to that file depending on the intended use of your boot file.
Do not forget that your default export needs to be a function. However, you can have as many named exports as you want, should the boot file expose something for later usage. In this case, you can import any of these named exports anywhere in your app.
The last step is to tell Quasar to use your new boot file. For this to happen you need to add the file in /quasar.conf.js
boot: [
// references /src/boot/<name>.js
'<name>'
]
When building a SSR app, you may want some boot files to run only on the server or only on the client, in which case you can do so like below:
boot: [
{
server: false, // run on client-side only!
path: '<name>' // references /src/boot/<name>.js
},
{
client: false, // run on server-side only!
path: '<name>' // references /src/boot/<name>.js
}
]
In case you want to specify boot files from node_modules, you can do so by prepending the path with ~ (tilde) character:
boot: [
// boot file from an npm package
'~my-npm-package/some/file'
]
If you want a boot file to be injected into your app only for a specific build type:
boot: [
ctx.mode.electron ? 'some-file' : ''
]
#
Redirecting to another page
#
WARNING
Please be mindful when redirecting as you might configure the app to go into an infinite redirect loop.
export default ({ urlPath, redirect }) => {
// ...
const isAuthorized = // ...
if (!isAuthorized && !urlPath.startsWith('/login')) {
redirect({ path: '/login' })
}
// ...
}
The redirect()
method accepts a String (full URL) or a Vue Router location String or Object. On SSR it can receive a second parameter which should be a Number for any of the HTTP STATUS codes that redirect the browser (3xx ones).
// Examples for redirect() with a Vue Router location:
redirect('/1') // Vue Router location as String
redirect({ path: '/1' }) // Vue Router location as Object
// Example for redirect() with a URL:
redirect('https://quasar.dev')
IMPORTANT!
The Vue Router location (in String or Object form) does not refer to URL path (and hash), but to the actual Vue Router routes that you have defined. So don’t add the publicPath to it and if you’re using the Vue Router hash mode then don’t add the hash to it.
- Let’s say that we have this Vue Router route defined:
{
path: '/one',
component: PageOne
}
Then regardless of our publicPath we can call redirect()
like this:
// publicPath: /wiki; vueRouterMode: history
redirect('/one') // good way
redirect({ path: '/one' }) // good way
redirect('/wiki/one') // WRONG!
// publicPath: /wiki; vueRouterMode: hash
redirect('/one') // good way
redirect({ path: '/one' }) // good way
redirect('/wiki/#/one') // WRONG!
// no publicPath; vueRouterMode: hash
redirect('/one') // good way
redirect({ path: '/one' }) // good way
redirect('/#/one') // WRONG!
// publicPath: /wiki; vueRouterMode: history
redirect('/one') // good way
redirect({ path: '/one' }) // good way
redirect('/wiki/one') // WRONG!
// publicPath: /wiki; vueRouterMode: hash
redirect('/one') // good way
redirect({ path: '/one' }) // good way
redirect('/wiki/#/one') // WRONG!
// no publicPath; vueRouterMode: hash
redirect('/one') // good way
redirect({ path: '/one' }) // good way
redirect('/#/one') // WRONG!
As it was mentioned in the previous sections, the default export of a boot file can return a Promise. If this Promise gets rejected with an Object that contains a “url” property, then Quasar CLI will redirect the user to that URL
:
export default ({ urlPath }) => {
return new Promise((resolve, reject) => {
// ...
const isAuthorized = // ...
if (!isAuthorized && !urlPath.startsWith('/login')) {
// the "url" param here is of the same type
// as for "redirect" above
reject({ url: '/login' })
return
}
// ...
})
}
Or a simpler equivalent:
export default () => {
// ...
const isAuthorized = // ...
if (!isAuthorized && !urlPath.startsWith('/login')) {
return Promise.reject({ url: '/login' })
}
// ...
}
#
Quasar App Flow
In order to better understand how a boot file works and what it does, you need to understand how your website/app boots:
Quasar
is initialized (components
,directives
,plugins
,Quasar
i18n
,Quasar
Quasar
Extras get imported (Roboto font
– ifused
, `Quasar
CSS
& your app’sglobal CSS
are importedApp.vue
is loaded (not yet being used)Store
is imported (if usingVuex
Store insrc/store
)Router
is imported (insrc/router
)Boot
files are importedRouter
default export function executedBoot
files get their default export function executed- (if on
Electron
mode)Electron
isimported
and injected intoVue
prototype - (if on
Cordova
mode) Listening for“deviceready”
event and only then continuing with following steps - Instantiating
Vue
withroot
component and attaching toDOM
#
Examples of boot files
#
Axios
import { boot } from 'quasar/wrappers'
import axios from 'axios'
const api = axios.create({ baseURL: 'https://api.example.com' })
export default boot(({ app }) => {
// for use inside Vue files (Options API) through this.$axios and this.$api
app.config.globalProperties.$axios = axios
// ^ ^ ^ this will allow you to use this.$axios (for Vue Options API form)
// so you won't necessarily have to import axios in each vue file
app.config.globalProperties.$api = api
// ^ ^ ^ this will allow you to use this.$api (for Vue Options API form)
// so you can easily perform requests against your app's API
})
export { axios, api }
#
vue-i18n
import { createI18n } from 'vue-i18n'
import messages from 'src/i18n'
export default ({ app }) => {
// Create I18n instance
const i18n = createI18n({
locale: 'en-US',
messages
})
// Tell app to use the I18n instance
app.use(i18n)
}
#
Router authentication
Some boot files might need to interfere with Vue Router configuration:
export default boot(({ router, store }) => {
router.beforeEach((to, from, next) => {
// Now you need to add your authentication logic here, like calling an API endpoint
})
})
#
Accessing data from boot files
Sometimes you want to access data that you configure in your boot
file in files where you don’t have access
to the root
Vue
instance.
Fortunately, because boot
files are just normal JavaScript files you can add as many named exports to your boot
file as you want.
Let’s take
the example of Axios
. Sometimes you want to access your Axios
instance inside your JavaScript
files, but you cannot access the root Vue instance. To solve this you can export the Axios instance in your boot file and import it elsewhere.
Consider the following boot
file for axios
:
// axios boot file (src/boot/axios.js)
import axios from 'axios'
// We create our own axios instance and set a custom base URL.
// Note that if we wouldn't set any config here we do not need
// a named export, as we could just `import axios from 'axios'`
const api = axios.create({
baseURL: 'https://api.example.com'
})
// for use inside Vue files through this.$axios and this.$api
// (only in Vue Options API form)
export default ({ app }) => {
app.config.globalProperties.$axios = axios
app.config.globalProperties.$api = api
}
// Here we define a named export
// that we can later use inside .js files:
export { axios, api }
In any JavaScript file, you’ll be able to import the axios instance like this.
// we import one of the named exports from src/boot/axios.js
import { api } from 'boot/axios'