mirror of https://github.com/vuejs/vitepress
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
257 lines
7.1 KiB
257 lines
7.1 KiB
3 years ago
|
# Using Vue in Markdown
|
||
|
|
||
2 years ago
|
In VitePress, each Markdown file is compiled into HTML and then processed as a [Vue Single-File Component](https://vuejs.org/guide/scaling-up/sfc.html). This means you can use any Vue features inside the Markdown, including dynamic templating, using Vue components, or arbitrary in-page Vue component logic by adding a `<script>` tag.
|
||
3 years ago
|
|
||
2 years ago
|
It's worth noting that VitePress leverages Vue's compiler to automatically detect and optimize the purely static parts of the Markdown content. Static contents are optimized into single placeholder nodes and eliminated from the page's JavaScript payload for initial visits. They are also skipped during client-side hydration. In short, you only pay for the dynamic parts on any given page.
|
||
3 years ago
|
|
||
2 years ago
|
::: tip SSR Compatibility
|
||
2 years ago
|
All Vue usage needs to be SSR-compatible. See [SSR Compatibility](./ssr-compat) for details and common workarounds.
|
||
|
:::
|
||
|
|
||
3 years ago
|
## Templating
|
||
|
|
||
|
### Interpolation
|
||
|
|
||
|
Each Markdown file is first compiled into HTML and then passed on as a Vue component to the Vite process pipeline. This means you can use Vue-style interpolation in text:
|
||
|
|
||
|
**Input**
|
||
|
|
||
|
```md
|
||
|
{{ 1 + 1 }}
|
||
|
```
|
||
|
|
||
|
**Output**
|
||
|
|
||
|
<div class="language-text"><pre><code>{{ 1 + 1 }}</code></pre></div>
|
||
|
|
||
|
### Directives
|
||
|
|
||
2 years ago
|
Directives also work (note that by design, raw HTML is also valid in Markdown):
|
||
3 years ago
|
|
||
|
**Input**
|
||
|
|
||
3 years ago
|
```html
|
||
|
<span v-for="i in 3">{{ i }}</span>
|
||
3 years ago
|
```
|
||
|
|
||
|
**Output**
|
||
|
|
||
|
<div class="language-text"><pre><code><span v-for="i in 3">{{ i }} </span></code></pre></div>
|
||
|
|
||
2 years ago
|
## `<script>` and `<style>`
|
||
3 years ago
|
|
||
2 years ago
|
Root-level `<script>` and `<style>` tags in Markdown files work just like they do in Vue SFCs, including `<script setup>`, `<style module>`, etc. The main difference here is that there is no `<template>` tag: all other root-level content is Markdown. Also note that all tags should be placed **after** the frontmatter:
|
||
2 years ago
|
|
||
|
```html
|
||
2 years ago
|
---
|
||
|
hello: world
|
||
|
---
|
||
|
|
||
2 years ago
|
<script setup>
|
||
|
import { ref } from 'vue'
|
||
|
|
||
|
const count = ref(0)
|
||
|
</script>
|
||
|
|
||
|
## Markdown Content
|
||
|
|
||
|
The count is: {{ count }}
|
||
|
|
||
2 years ago
|
<button :class="$style.button" @click="count++">Increment</button>
|
||
2 years ago
|
|
||
2 years ago
|
<style module>
|
||
|
.button {
|
||
2 years ago
|
color: red;
|
||
|
font-weight: bold;
|
||
|
}
|
||
|
</style>
|
||
|
```
|
||
|
|
||
2 years ago
|
::: warning Avoid `<style scoped>` in Markdown
|
||
2 years ago
|
When used in Markdown, `<style scoped>` requires adding special attributes to every element on the current page, which will significantly bloat the page size. `<style module>` is preferred when locally-scoped styling is needed in a page.
|
||
|
:::
|
||
|
|
||
2 years ago
|
You also have access to VitePress' runtime APIs such as the [`useData` helper](../reference/runtime-api#usedata), which provides access to current page's metadata:
|
||
3 years ago
|
|
||
|
**Input**
|
||
|
|
||
3 years ago
|
```html
|
||
3 years ago
|
<script setup>
|
||
|
import { useData } from 'vitepress'
|
||
|
|
||
|
const { page } = useData()
|
||
|
</script>
|
||
|
|
||
|
<pre>{{ page }}</pre>
|
||
|
```
|
||
|
|
||
|
**Output**
|
||
|
|
||
|
```json
|
||
|
{
|
||
|
"path": "/using-vue.html",
|
||
|
"title": "Using Vue in Markdown",
|
||
2 years ago
|
"frontmatter": {},
|
||
|
...
|
||
3 years ago
|
}
|
||
|
```
|
||
|
|
||
|
## Using Components
|
||
|
|
||
2 years ago
|
You can import and use Vue components directly in Markdown files.
|
||
3 years ago
|
|
||
2 years ago
|
### Importing in Markdown
|
||
3 years ago
|
|
||
2 years ago
|
If a component is only used by a few pages, it's recommended to explicitly import them where they are used. This allows them to be properly code-split and only loaded when the relevant pages are shown:
|
||
3 years ago
|
|
||
|
```md
|
||
|
<script setup>
|
||
|
import CustomComponent from '../components/CustomComponent.vue'
|
||
|
</script>
|
||
|
|
||
|
# Docs
|
||
|
|
||
|
This is a .md using a custom component
|
||
|
|
||
|
<CustomComponent />
|
||
|
|
||
|
## More docs
|
||
|
|
||
|
...
|
||
|
```
|
||
|
|
||
2 years ago
|
### Registering Components Globally
|
||
3 years ago
|
|
||
2 years ago
|
If a component is going to be used on most of the pages, they can be registered globally by customizing the Vue app instance. See relevant section in [Extending Default Theme](./extending-default-theme#registering-global-components) for an example.
|
||
3 years ago
|
|
||
|
::: warning IMPORTANT
|
||
2 years ago
|
Make sure a custom component's name either contains a hyphen or is in PascalCase. Otherwise, it will be treated as an inline element and wrapped inside a `<p>` tag, which will lead to hydration mismatch because `<p>` does not allow block elements to be placed inside it.
|
||
3 years ago
|
:::
|
||
|
|
||
|
### Using Components In Headers <ComponentInHeader />
|
||
|
|
||
|
You can use Vue components in the headers, but note the difference between the following syntaxes:
|
||
|
|
||
|
| Markdown | Output HTML | Parsed Header |
|
||
|
| ------------------------------------------------------- | ----------------------------------------- | ------------- |
|
||
|
| <pre v-pre><code> # text <Tag/> </code></pre> | `<h1>text <Tag/></h1>` | `text` |
|
||
|
| <pre v-pre><code> # text \`<Tag/>\` </code></pre> | `<h1>text <code><Tag/></code></h1>` | `text <Tag/>` |
|
||
|
|
||
|
The HTML wrapped by `<code>` will be displayed as-is; only the HTML that is **not** wrapped will be parsed by Vue.
|
||
|
|
||
|
::: tip
|
||
2 years ago
|
The output HTML is accomplished by [Markdown-it](https://github.com/Markdown-it/Markdown-it), while the parsed headers are handled by VitePress (and used for both the sidebar and document title).
|
||
|
:::
|
||
|
|
||
|
|
||
|
## Escaping
|
||
|
|
||
|
You can escape Vue interpolations by wrapping them in a `<span>` or other elements with the `v-pre` directive:
|
||
|
|
||
|
**Input**
|
||
|
|
||
|
```md
|
||
|
This <span v-pre>{{ will be displayed as-is }}</span>
|
||
|
```
|
||
|
|
||
|
**Output**
|
||
|
|
||
|
<div class="escape-demo">
|
||
|
<p>This <span v-pre>{{ will be displayed as-is }}</span></p>
|
||
|
</div>
|
||
|
|
||
|
Alternatively, you can wrap the entire paragraph in a `v-pre` custom container:
|
||
|
|
||
|
```md
|
||
|
::: v-pre
|
||
2 years ago
|
{{ This will be displayed as-is }}
|
||
2 years ago
|
:::
|
||
|
```
|
||
|
|
||
|
**Output**
|
||
|
|
||
|
<div class="escape-demo">
|
||
|
|
||
|
::: v-pre
|
||
|
{{ This will be displayed as-is }}
|
||
3 years ago
|
:::
|
||
|
|
||
2 years ago
|
</div>
|
||
|
|
||
|
## Unescape in Code Blocks
|
||
|
|
||
2 years ago
|
By default, all fenced code blocks are automatically wrapped with `v-pre`, so no Vue syntax will be processed inside. To enable Vue-style interpolation inside fences, you can append the language with the `-vue` suffix, e.g. `js-vue`:
|
||
2 years ago
|
|
||
|
**Input**
|
||
|
|
||
|
````md
|
||
|
```js-vue
|
||
|
Hello {{ 1 + 1 }}
|
||
|
```
|
||
|
````
|
||
|
|
||
|
**Output**
|
||
|
|
||
|
```js-vue
|
||
|
Hello {{ 1 + 1 }}
|
||
|
```
|
||
|
|
||
2 years ago
|
Note that this might prevent certain tokens from being syntax highlighted properly.
|
||
|
|
||
3 years ago
|
## Using CSS Pre-processors
|
||
|
|
||
|
VitePress has [built-in support](https://vitejs.dev/guide/features.html#css-pre-processors) for CSS pre-processors: `.scss`, `.sass`, `.less`, `.styl` and `.stylus` files. There is no need to install Vite-specific plugins for them, but the corresponding pre-processor itself must be installed:
|
||
|
|
||
|
```
|
||
|
# .scss and .sass
|
||
|
npm install -D sass
|
||
|
|
||
|
# .less
|
||
|
npm install -D less
|
||
|
|
||
|
# .styl and .stylus
|
||
|
npm install -D stylus
|
||
|
```
|
||
|
|
||
|
Then you can use the following in Markdown and theme components:
|
||
|
|
||
|
```vue
|
||
|
<style lang="sass">
|
||
|
.title
|
||
|
font-size: 20px
|
||
|
</style>
|
||
|
```
|
||
|
|
||
2 years ago
|
## Using Teleports
|
||
|
|
||
7 months ago
|
VitePress currently has SSG support for teleports to body only. For other targets, you can wrap them inside the built-in `<ClientOnly>` component or inject the teleport markup into the correct location in your final page HTML through [`postRender` hook](../reference/site-config#postrender).
|
||
2 years ago
|
|
||
|
<ModalDemo />
|
||
|
|
||
|
::: details
|
||
|
<<< @/components/ModalDemo.vue
|
||
|
:::
|
||
|
|
||
|
```md
|
||
|
<ClientOnly>
|
||
|
<Teleport to="#modal">
|
||
|
<div>
|
||
|
// ...
|
||
|
</div>
|
||
|
</Teleport>
|
||
|
</ClientOnly>
|
||
|
```
|
||
|
|
||
|
<script setup>
|
||
7 months ago
|
import ModalDemo from '../../components/ModalDemo.vue'
|
||
|
import ComponentInHeader from '../../components/ComponentInHeader.vue'
|
||
2 years ago
|
</script>
|
||
2 years ago
|
|
||
|
<style>
|
||
|
.escape-demo {
|
||
|
border: 1px solid var(--vp-c-border);
|
||
|
border-radius: 8px;
|
||
|
padding: 0 20px;
|
||
|
}
|
||
|
</style>
|