Usando Vue en Markdown
En VitePress, cada archivo Markdown es compilado para HTML y entonces procesado como un Componente de Archivo Único de Vue. Esto significa que puede usar cualquier funcionalidad de Vue dentro del Markdown, incluyendo la interpolación dinámica, usar componentes Vue o lógica arbitrária de componentes Vue dentro de la página adicionando una tag <script>
.
Vale resaltar que VitePress aprovecha el compilador Vue para detectar y optimizar automáticamente las partes puramente estáticas del contenido Markdown. Los contenidos estáticaos son optimizados en nodos de espacio reservado únicos y eliminados de la carga JavaScript de la página para visitas iniciales. Ellos también son ignorados durante la hidratación en el lado del cliente. En resumen, solo paga por las partes dinámicas en cualquier página específica.
Compatibilidad SSR
Todo uso de Vue necesita ser compatible con SSR. Consulte Compatibilidad SSR para detalles y soluciones comunes.
Creación de Templates
Interpolación
Cada archivo Markdown es primero compilado para HTML y después pasado como un componente Vue para la canalización de procesos Vite. Esto significa que puede usar interpolación en el estilo Vue en el texto:
Entrada
{{ 1 + 1 }}
Salida
2
Directivas
Las Directivas también funcionan (observe que, por definición, HTML crudo también es válido en Markdown):
Entrada
<span v-for="i in 3">{{ i }}</span>
Salida
1 2 3
<script>
e <style>
las tags <script>
e <style>
en nivel raiz en los archivos Markdown funcionan igualmente como en los componentes de archivo único Vue, incluyendo <script setup>
, <style module>
, y etc. La principal diferencia aquí es que no hay una tag <template>
: todo contenido en nivel raiz es Markdown. Además, observe que todas las tags deben ser colocadas después del frontmatter:
---
hello: world
---
<script setup>
import { ref } from 'vue'
const count = ref(0)
</script>
## Contenido Markdown
El conteo es: {{ count }}
<button :class="$style.button" @click="count++">Incrementar</button>
<style module>
.button {
color: red;
font-weight: bold;
}
</style>
Evite <style scoped>
en el Markdown
Cuando es usado en Markdown, <style scoped>
exige la adición de atributos especiales a cada elemento en la página actual, lo que aumentará significativamente el tamaño de la página. <style module>
es preferido cuando es necesaria una estilización localizada en una página.
También tiene acceso a los APIs de tiempo de ejecución VitePress, como el auxiliar useData
, que proporciona acceso a los metadados de la página actual:
Entrada
<script setup>
import { useData } from 'vitepress'
const { page } = useData()
</script>
<pre>{{ page }}</pre>
Salida
{
"path": "/usando-vue.html",
"title": "Usando Vue en Markdown",
"frontmatter": {},
...
}
Usando Componentes
Puede importar y usar componentes Vue directamente en los archivos Markdown.
Importando en el Markdown
Si un componente es usado apenas por algunas páginas, es recomendable importarlos explicitamente donde son usados. Esto permite que ellos sean divididos adecuadamente y cargados apenas cuando las páginas relevantes son mostradas.
<script setup>
import CustomComponent from '../components/CustomComponent.vue'
</script>
# Documentación
Este es un archivo .md usando un componente personalizado
<CustomComponent />
## Más documentación
...
Registrando Componentes Globalmente
Si un componente fuera usado en la mayoría de las páginas, ellos pueden ser registrados globalmente personalizando la instancia de la aplicación Vue. Consulte la sección relevante en Extendiendo el Tema por Defecto para un ejemplo.
IMPORTANT
Asegurese de que el nombre de un componente personalizado contenga un hífen o esté en PascalCase. Caso contrario, el será tratado como un elemento alineado y envuelto dentro de una tag <p>
, lo que llevará a una incompatibilidad de hidratación pues <p>
no permite que elementos de bloque sean colocados dentro de el.
Usando Componentes En Headers
Puede usar componentes Vue en los headers, pero observe la diferencia entre las siguientes sintaxis:
Markdown | HTML de Salida | Header Procesado |
---|---|---|
| <h1>texto <Tag/></h1> | texto |
| <h1>texto <code><Tag/></code></h1> | texto <Tag/> |
El HTML envuelto por <code>
será mostrado como es, solamente el HTML que no estuviera envuelto será analizado por Vue.
TIP
EL HTML de salida es realizado por Markdown-it, en cuanto los headers procesados son manipulados por VitePress (y usados tanto en la barra lateral como dentro del título del video).
Escapes
Puede escapar de interpolaciones Vue envolvientdolas en un <span>
u otros elementos con la directiva v-pre
:
Entrada
Esto <span v-pre>{{ será exibido como es }}</span>
Salida
Esto {{ será exibido como es }}
Alternativamente, puede envolver todo el paragrafo en un contenedor personalizadon v-pre
:
::: v-pre
{{ Esto será exibido como es }}
:::
Output
{{ Esto será exibido como es }}
"Desescape" en bloques de Código
Por defecto, todos los bloques de código cercados son automáticamente envueltos con v-pre
, entonces ninguna sintaxis Vue será procesada dentro de ellos. Para permitir la interpolación en el estilo Vue dentro de la valla, puede adicionar el lenguaje con el sufijo -vue
, por ejemplo, js-vue
:
Entrada
```js-vue
Hola {{ 1 + 1 }}
```
Salida
Hola 2
Observe que esto puede impedir que ciertos tokens sean realzados correctamente.
Usando Preprocesadores CSS
VitePress poseé soporte embutido para preprocesadores CSS: archivos .scss
, .sass
, .less
, .styl
e .stylus
. No es necesario instalar plugins específicos de Vite para ellos, pero el propio preprocesados correspondiente debe ser instalado:
# .scss e .sass
npm install -D sass
# .less
npm install -D less
# .styl e .stylus
npm install -D stylus
Entonces puede usar lo siguiente en Markdown y en los componentes del tema:
<style lang="sass">
.title
font-size: 20px
</style>
Usando Teleports
VitePress actualmente ofrece soporte a SSG para teleports apenas para el cuerpo. Para otros objetivos, puede envolverlos dentro del componente embutido <ClientOnly>
o inyectar la marcación de teleport en la localización correcta en su página final HTML por medio del hook postRender
.
Details
<script setup lang="ts">
import { ref } from 'vue'
const showModal = ref(false)
</script>
<template>
<button class="modal-button" @click="showModal = true">Show Modal</button>
<Teleport to="body">
<Transition name="modal">
<div v-show="showModal" class="modal-mask">
<div class="modal-container">
<p>Hello from the modal!</p>
<div class="model-footer">
<button class="modal-button" @click="showModal = false">
Close
</button>
</div>
</div>
</div>
</Transition>
</Teleport>
</template>
<style scoped>
.modal-mask {
position: fixed;
z-index: 200;
top: 0;
left: 0;
width: 100%;
height: 100%;
background-color: rgba(0, 0, 0, 0.5);
display: flex;
align-items: center;
justify-content: center;
transition: opacity 0.3s ease;
}
.modal-container {
width: 300px;
margin: auto;
padding: 20px 30px;
background-color: var(--vp-c-bg);
border-radius: 2px;
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.33);
transition: all 0.3s ease;
}
.model-footer {
margin-top: 8px;
text-align: right;
}
.modal-button {
padding: 4px 8px;
border-radius: 4px;
border-color: var(--vp-button-alt-border);
color: var(--vp-button-alt-text);
background-color: var(--vp-button-alt-bg);
}
.modal-button:hover {
border-color: var(--vp-button-alt-hover-border);
color: var(--vp-button-alt-hover-text);
background-color: var(--vp-button-alt-hover-bg);
}
.modal-enter-from,
.modal-leave-to {
opacity: 0;
}
.modal-enter-from .modal-container,
.modal-leave-to .modal-container {
transform: scale(1.1);
}
</style>
<ClientOnly>
<Teleport to="#modal">
<div>
// ...
</div>
</Teleport>
</ClientOnly>