Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/JohnCampionJr/vite-plugin-vue-layouts
Vue layout plugin for Vite
https://github.com/JohnCampionJr/vite-plugin-vue-layouts
Last synced: about 1 month ago
JSON representation
Vue layout plugin for Vite
- Host: GitHub
- URL: https://github.com/JohnCampionJr/vite-plugin-vue-layouts
- Owner: JohnCampionJr
- License: mit
- Created: 2021-02-11T22:23:06.000Z (almost 4 years ago)
- Default Branch: main
- Last Pushed: 2024-02-22T01:02:37.000Z (10 months ago)
- Last Synced: 2024-11-03T08:21:10.501Z (about 1 month ago)
- Language: TypeScript
- Size: 579 KB
- Stars: 473
- Watchers: 4
- Forks: 65
- Open Issues: 15
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE
Awesome Lists containing this project
- awesome-vite - `vite-plugin-vue-layouts` - layouts for pages (Pre-packed / Plugins)
README
# vite-plugin-vue-layouts
[![npm version](https://img.shields.io/npm/v/vite-plugin-vue-layouts)](https://www.npmjs.com/package/vite-plugin-vue-layouts)
> Router based layout for Vue 3 applications using [Vite](https://github.com/vitejs/vite)
## Overview
This works best along with the [vite-plugin-pages](https://github.com/hannoeru/vite-plugin-pages).
Layouts are stored in the `/src/layouts` folder by default and are standard Vue components with a `` in the template.
Pages without a layout specified use `default.vue` for their layout.
You can use route blocks to allow each page to determine its layout. The block below in a page will look for `/src/layouts/users.vue` for its layout.
See the [Vitesse starter template](https://github.com/antfu/vitesse) for a working example.
```html
meta:
layout: users```
## Getting Started
Install Layouts:
```bash
$ npm install -D vite-plugin-vue-layouts
```Add to your `vite.config.js`:
```js
import Vue from '@vitejs/plugin-vue';
import Pages from 'vite-plugin-pages';
import Layouts from 'vite-plugin-vue-layouts';export default {
plugins: [Vue(), Pages(), Layouts()],
};
```In main.ts, you need to add a few lines to import the generated code and setup the layouts.
## vue-router
```js
import { createRouter } from 'vue-router'
import { setupLayouts } from 'virtual:generated-layouts'
import generatedRoutes from '~pages'const routes = setupLayouts(generatedRoutes)
const router = createRouter({
// ...
routes,
});
```## [unplugin-vue-router](https://github.com/posva/unplugin-vue-router)
```js
import { createRouter } from 'vue-router/auto'
import { setupLayouts } from 'virtual:generated-layouts'const router = createRouter({
// ...
extendRoutes: (routes) => setupLayouts(routes),
})
```## Client Types
If you want type definition of `virtual:generated-layouts`, add `vite-plugin-vue-layouts/client` to `compilerOptions.types` of your `tsconfig`:
```json
{
"compilerOptions": {
"types": ["vite-plugin-vue-layouts/client"]
}
}
```## Configuration
```ts
interface UserOptions {
layoutsDirs?: string | string[]
pagesDirs?: string | string[] | null
exclude: string[]
defaultLayout?: string
}
```### Using configuration
To use custom configuration, pass your options to Layouts when instantiating the plugin:
```js
// vite.config.js
import Layouts from 'vite-plugin-vue-layouts';export default {
plugins: [
Layouts({
layoutsDirs: 'src/mylayouts',
pagesDirs: 'src/pages',
defaultLayout: 'myDefault'
}),
],
};
```### layoutsDirs
Relative path to the layouts directory. Supports globs.
All .vue files in this folder are imported async into the generated code.Can also be an array of layout dirs
Can use `**` to support scenarios like `module1/layouts` and `modules2/layouts` with a setting of `src/**/layouts`
Any files named `__*__.vue` will be excluded, and you can specify any additional exclusions with the `exclude` option
**Default:** `'src/layouts'`
### pagesDirs
Defines the pages dir to avoid HMR reloading for all added or deleted files anywhere in the project.
Relative path to the pages directory. If you want it to watch for all files, like in v0.8.0 or earlier, set to null.
Can also be an array of layout dirs or use `**` glob patterns
**Default:** `'src/pages'`
## How it works
`setupLayouts` transforms the original `router` by
1. Replacing every page with its specified layout
2. Appending the original page in the `children` property.Simply put, layouts are [nested routes](https://next.router.vuejs.org/guide/essentials/nested-routes.html#nested-routes) with the same path.
Before:
```
router: [ page1, page2, page3 ]
```After `setupLayouts()`:
```
router: [
layoutA: page1,
layoutB: page2,
layoutA: page3,
]
```That means you have the full flexibility of the [vue-router API](https://next.router.vuejs.org/api/) at your disposal.
## Common patterns
### Transitions
Layouts and Transitions work as expected and explained in the [vue-router docs](https://next.router.vuejs.org/guide/advanced/transitions.html) only as long as `Component` changes on each route. So if you want a transition between pages with the same layout *and* a different layout, you have to mutate `:key` on `` (for a detailed example, see the vue docs about [transitions between elements](https://v3.vuejs.org/guide/transitions-enterleave.html#transitioning-between-elements)).`App.vue`
```html
```
Now Vue will always trigger a transition if you change the route.
### Data from layout to page
If you want to send data *down* from the layout to the page, use props
``````
### Set static data at the page
If you want to set state in your page and do something with it in your layout, add additional properties to a route's `meta` property. Doing so only works if you know the state at build-time.
You can use the `` block if you work with [vite-plugin-pages](https://github.com/hannoeru/vite-plugin-pages).
In `page.vue`:
```htmlContentmeta:
layout: default
bgColor: yellow```
Now you can read `bgColor` in `layout.vue`:
```htmlimport { useRouter } from 'vue-router'
```
### Data dynamically from page to layout
If you need to set `bgColor` dynamically at run-time, you can use [custom events](https://v3.vuejs.org/guide/component-custom-events.html#custom-events).
Emit the event in `page.vue`:
```htmlimport { defineEmit } from 'vue'
const emit = defineEmit(['setColor'])if (2 + 2 === 4)
emit('setColor', 'green')
else
emit('setColor', 'red')```
Listen for `setColor` custom-event in `layout.vue`:
```htmlimport { ref } from 'vue'
const bgColor = ref('yellow')
const setBg = (color) => {
bgColor.value = color
}
```
## ClientSideLayout
The clientSideLayout uses a simpler [virtual file](https://vitejs.dev/guide/api-plugin.html#importing-a-virtual-file) + [glob import](https://vitejs.dev/guide/features.html#glob-import) scheme, This means that its hmr is faster and more accurate, but also more limited
### Usage
```js
// vite.config.js
import { ClientSideLayout } from 'vite-plugin-vue-layouts'export default {
plugins: [
ClientSideLayout({
layoutsDir: 'src/mylayouts', // default to 'src/layouts'
defaultLayout: 'myDefault', // default to 'default', no need '.vue'
importMode: 'sync' // The default will automatically detect -> ssg is sync,other is async
}),
],
};
```