Extending a Theme
Sometimes you might want make some minor changes to a theme, but you may not want to fork and modify the entire project.
With the help of Theme API, you can extend a theme and make your own modifications:
const { path } = require('@vuepress/utils')
module.exports = {
// your theme
name: 'vuepress-theme-foo',
// parent theme to extend
extends: 'vuepress-theme-bar',
// override layouts of parent theme
layouts: {
Layout: path.resolve(__dirname, 'layouts/Layout.vue'),
},
}
2
3
4
5
6
7
8
9
10
11
12
import type { ThemeObject } from '@vuepress/core'
import { path } from '@vuepress/utils'
const fooTheme: ThemeObject = {
// your theme
name: 'vuepress-theme-foo',
// parent theme to extend
extends: 'vuepress-theme-bar',
// override layouts of parent theme
layouts: {
Layout: path.resolve(__dirname, 'layouts/Layout.vue'),
},
}
export default fooTheme
2
3
4
5
6
7
8
9
10
11
12
13
14
15
In this case, your vuepress-theme-foo
will inherit all configuration, plugins and layouts from vuepress-theme-bar
, and you can override corresponding layouts as needed.
Extend Default Theme
First, create the theme directory and the theme entry .vuepress/theme/index.js
:
const { path } = require('@vuepress/utils')
module.exports = {
name: 'vuepress-theme-local',
extends: '@vuepress/theme-default',
layouts: {
Layout: path.resolve(__dirname, 'layouts/Layout.vue'),
},
}
2
3
4
5
6
7
8
9
import type { ThemeObject } from '@vuepress/core'
import { path } from '@vuepress/utils'
const localTheme: ThemeObject = {
name: 'vuepress-theme-local',
extends: '@vuepress/theme-default',
layouts: {
Layout: path.resolve(__dirname, 'layouts/Layout.vue'),
},
}
export default localTheme
2
3
4
5
6
7
8
9
10
11
12
You local theme will extends default theme, and override the Layout
layout.
Next, create .vuepress/theme/layouts/Layout.vue
, and make use of the slots that provided by the Layout
of default theme:
<template>
<Layout>
<template #page-bottom>
<div class="my-footer">This is my custom page footer</div>
</template>
</Layout>
</template>
<script>
import Layout from '@vuepress/theme-default/lib/client/layouts/Layout.vue'
export default {
components: {
Layout,
},
}
</script>
<style lang="css">
.my-footer {
text-align: center;
}
</style>
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
You will add a custom footer to every normal pages in default theme (excluding homepage):
Here are all the slots that provided by the Layout
of default theme:
navbar
navbar-before
navbar-after
sidebar
sidebar-top
sidebar-bottom
page
page-top
page-bottom
Finally, remember to use your local theme in .vuepress/config.js
:
const { path } = require('@vuepress/utils')
module.exports = {
theme: path.resolve(__dirname, './theme'),
}
2
3
4
5
import { path } from '@vuepress/utils'
import { defineUserConfig } from 'vuepress'
import type { DefaultThemeOptions } from 'vuepress'
export default defineUserConfig<DefaultThemeOptions>({
theme: path.resolve(__dirname, './theme'),
})
2
3
4
5
6
7
Make Your Theme Extendable
As a theme author, you might want to make your theme extendable, allowing users to use your theme with their own customization.
You can provide slots in your layouts, just like how default theme does. This approach requires you to determine which parts of your theme could be extended. It is more suitable for those common customizations like page footer or header:
<template>
<div class="my-theme-layout">
<slot name="page-header" />
<Content />
<slot name="page-footer" />
</div>
</template>
2
3
4
5
6
7
If you think it is not flexible enough, you can try some more aggressive approaches to make each components of you theme replaceable.
For example, set alias
for each components of you theme:
module.exports = {
name: 'vuepress-theme-foo',
alias: {
// set alias for replaceable components
'@theme/Navbar.vue': path.resolve(__dirname, 'components/Navbar.vue'),
'@theme/Sidebar.vue': path.resolve(__dirname, 'components/Sidebar.vue'),
},
}
2
3
4
5
6
7
8
Next, use those components via aliases in your theme:
<template>
<div class="my-theme-layout">
<Navbar />
<Sidebar />
<Content />
</div>
</template>
<script>
import Navbar from '@theme/Navbar.vue'
import Sidebar from '@theme/Sidebar.vue'
export default {
components: {
Navbar,
Sidebar,
},
}
</script>
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
Then, users can replace specific components by overriding the alias
when extending your theme:
module.exports = {
name: 'vuepress-theme-foobar',
extends: 'vuepress-theme-foo'
alias: {
// replace the Navbar component
'@theme/Navbar.vue': path.resolve(__dirname, 'components/CustomNavbar.vue'),
},
}
2
3
4
5
6
7
8