Routing
File-Based Routing
VitePress uses file-based routing, which means the generated HTML pages are mapped from the directory structure of the source Markdown files. For example, given the following directory structure:
Directoryguide/
- getting-started.md
- index.md
- index.md
- prologue.md
The generated HTML pages will be:
The resulting HTML can be hosted on any web server that can serve static files.
Root and Source Directory
There are two important concepts in the file structure of a VitePress project: the project root and the source directory.
Project Root
Project root is where VitePress will try to look for the .vitepress
special directory. The .vitepress
directory is a reserved location for VitePress’ config file, dev server cache, build output, and optional theme customization code.
When you run vitepress dev
or vitepress build
from the command line, VitePress will use the current working directory as project root. To specify a sub-directory as root, you will need to pass the relative path to the command. For example, if your VitePress project is located in ./docs
, you should run vitepress dev docs
:
Directorydocs/
- .vitepress
- getting-started.md
- index.md
- …
This is going to result in the following source-to-HTML mapping:
Source Directory
Source directory is where your Markdown source files live. By default, it is the same as the project root. However, you can configure it via the srcDir
config option.
The srcDir
option is resolved relative to project root. For example, with srcDir: 'src'
, your file structure will look like this:
Directory.vitepress/
- …
Directorysrc/
- getting-started.md
- index.md
The resulting source-to-HTML mapping:
Linking Between Pages
You can use both absolute and relative paths when linking between pages. Note that although both .md
and .html
extensions will work, the best practice is to omit file extensions so that VitePress can generate the final URLs based on your config.
Learn more about linking to assets such images in Asset Handling.
Linking to Non-VitePress Pages
If you want to link to a page in your site that is not generated by VitePress, you’ll either need to use the full URL (opens in a new tab) or explicitly specify the target:
Input
Output
Generating Clean URL
By default, VitePress resolves inbound links to URLs ending with .html
. However, some users may prefer “Clean URLs” without the .html
extension - for example, example.com/path
instead of example.com/path.html
.
Some servers or hosting platforms (for example Netlify, Vercel, GitHub Pages) provide the ability to map a URL like /foo
to /foo.html
if it exists, without a redirect:
- Netlify and GitHub Pages support this by default.
- Vercel requires enabling the
cleanUrls
option invercel.json
.
If this feature is available to you, you can then also enable VitePress’ own cleanUrls
config option so that:
- Inbound links between pages are generated without the
.html
extension. - If current path ends with
.html
, the router will perform a client-side redirect to the extension-less path.
If, however, you cannot configure your server with such support, you will have to manually resort to the following directory structure:
Directorygetting-started/
- index.md
Directoryinstallation/
- index.md
- index.md
Route Rewrites
You can customize the mapping between the source directory structure and the generated pages. It’s useful when you have a complex project structure. For example, let’s say you have a monorepo with multiple packages, and would like to place documentations along with the source files like this:
Directorypackages/
Directorypkg-a/
Directorysrc/
- pkg-a-code.ts
- pkg-a-docs.md
Directorypkg-b/
Directorysrc/
- pkg-b-code.ts
- pkg-b-docs.md
And you want the VitePress pages to be generated like this:
You can achieve this by configuring the rewrites
option like this:
The rewrites
option also supports dynamic route parameters. In the above example, it would be verbose to list all the paths if you have many packages. Given that they all have the same file structure, you can simplify the config like this:
The rewrite paths are compiled using the path-to-regexp
package - consult its documentation for more advanced syntax.
Dynamic Routes
You can generate many pages using a single Markdown file and dynamic data. For example, you can create a packages/[pkg].md
file that generates a corresponding page for every package in a project. Here, the [pkg]
segment is a route parameter that differentiates each page from the others.
Paths Loader File
Since VitePress is a static site generator, the possible page paths must be determined at build time. Therefore, a dynamic route page must be accompanied by a paths loader file. For packages/[pkg].md
, we will need packages/[pkg].paths.js
(.ts
is also supported):
Directorypackages/
- [pkg].md
- [pkg].paths.js
The paths loader should provide an object with a paths
method as its default export. The paths
method should return an array of objects with a params
property. Each of these objects will generate a corresponding page.
Given the following paths
array:
The generated HTML pages will be:
Directorypackages/
- foo.md
- bar.md
Multiple Params
A dynamic route can contain multiple params:
File Structure
Directorypackages/
- [pkg]-[version].md
- [pkg]-[version].paths.js
Paths Loader
Output
Directorypackages/
- foo-1.0.0.md
- foo-2.0.0.md
- bar-1.0.0.md
- bar-2.0.0.md
Dynamically Generating Paths
The paths loader module is run in Node.js and only executed during build time. You can dynamically generate the paths array using any data, either local or remote.
Generating paths from local files:
Generating paths from remote data:
Accessing Params in Page
You can use the params to pass additional data to each page. The Markdown route file can access the current page params in Vue expressions via the $params
global property:
You can also access the current page’s params via the useData
runtime API. This is available in both Markdown files and Vue components:
Rendering Raw Content
Params passed to the page will be serialized in the client JavaScript payload, so you should avoid passing heavy data in params, for example raw Markdown or HTML content fetched from a remote CMS.
Instead, you can pass such content to each page using the content
property on each path object:
Then, use the following special syntax to render the content as part of the Markdown file itself: