Skip to content

Commit 13f94a6

Browse files
zonemeenbrc-dd
andauthored
feat(build): custom excerpt for createContentLoader (vuejs#2698)
Co-authored-by: Divyansh Singh <40380293+brc-dd@users.noreply.github.com>
1 parent 68f25f5 commit 13f94a6

File tree

6 files changed

+87
-19
lines changed

6 files changed

+87
-19
lines changed

docs/guide/data-loading.md

+48-6
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@ The loader module is evaluated only in Node.js, so you can import Node APIs and
2323

2424
You can then import data from this file in `.md` pages and `.vue` components using the `data` named export:
2525

26-
```html
26+
```vue
2727
<script setup>
2828
import { data } from './example.data.js'
2929
</script>
@@ -70,7 +70,7 @@ export default {
7070
// watchedFiles will be an array of absolute paths of the matched files.
7171
// generate an array of blog post metadata that can be used to render
7272
// a list in the theme layout
73-
return watchedFiles.map(file => {
73+
return watchedFiles.map((file) => {
7474
return parse(fs.readFileSync(file, 'utf-8'), {
7575
columns: true,
7676
skip_empty_lines: true
@@ -147,9 +147,9 @@ export default createContentLoader('posts/*.md', {
147147
// the final result is what will be shipped to the client.
148148
return rawData.sort((a, b) => {
149149
return +new Date(b.frontmatter.date) - +new Date(a.frontmatter.date)
150-
}).map(page => {
151-
page.src // raw markdown source
152-
page.html // rendered full page HTML
150+
}).map((page) => {
151+
page.src // raw markdown source
152+
page.html // rendered full page HTML
153153
page.excerpt // rendered excerpt HTML (content above first `---`)
154154
return {/* ... */}
155155
})
@@ -159,7 +159,7 @@ export default createContentLoader('posts/*.md', {
159159

160160
Check out how it is used in the [Vue.js blog](https://github.com/vuejs/blog/blob/main/.vitepress/theme/posts.data.ts).
161161

162-
The `createContentLoader` API can also be used inside [build hooks](/reference/site-config#build-hooks):
162+
The `createContentLoader` API can also be used inside [build hooks](../reference/site-config#build-hooks):
163163

164164
```js
165165
// .vitepress/config.js
@@ -171,6 +171,48 @@ export default {
171171
}
172172
```
173173

174+
**Types**
175+
176+
```ts
177+
interface ContentOptions<T = ContentData[]> {
178+
/**
179+
* Include src?
180+
* @default false
181+
*/
182+
includeSrc?: boolean
183+
184+
/**
185+
* Render src to HTML and include in data?
186+
* @default false
187+
*/
188+
render?: boolean
189+
190+
/**
191+
* If `boolean`, whether to parse and include excerpt? (rendered as HTML)
192+
*
193+
* If `function`, control how the excerpt is extracted from the content.
194+
*
195+
* If `string`, define a custom separator to be used for extracting the
196+
* excerpt. Default separator is `---` if `excerpt` is `true`.
197+
*
198+
* @see https://github.com/jonschlinkert/gray-matter#optionsexcerpt
199+
* @see https://github.com/jonschlinkert/gray-matter#optionsexcerpt_separator
200+
*
201+
* @default false
202+
*/
203+
excerpt?:
204+
| boolean
205+
| ((file: { data: { [key: string]: any }; content: string; excerpt?: string }, options?: any) => void)
206+
| string
207+
208+
/**
209+
* Transform the data. Note the data will be inlined as JSON in the client
210+
* bundle if imported from components or markdown files.
211+
*/
212+
transform?: (data: ContentData[]) => T | Promise<T>
213+
}
214+
```
215+
174216
## Typed Data Loaders
175217

176218
When using TypeScript, you can type your loader and `data` export like so:

docs/guide/deploy.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -73,7 +73,7 @@ Cache-Control: max-age=31536000,immutable
7373
cache-control: immutable
7474
```
7575

76-
Note: the `_headers` file should be placed in the [public directory](/guide/asset-handling#the-public-directory) - in our case, `docs/public/_headers` - so that it is copied verbatim to the output directory.
76+
Note: the `_headers` file should be placed in the [public directory](./asset-handling#the-public-directory) - in our case, `docs/public/_headers` - so that it is copied verbatim to the output directory.
7777

7878
[Netlify custom headers documentation](https://docs.netlify.com/routing/headers/)
7979

docs/guide/extending-default-theme.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -59,10 +59,10 @@ export default DefaultTheme
5959
```
6060

6161
::: warning
62-
If you are using optional components like the [Team Page](/reference/default-theme-team-page) components, make sure to also import them from `vitepress/theme-without-fonts`!
62+
If you are using optional components like the [Team Page](../reference/default-theme-team-page) components, make sure to also import them from `vitepress/theme-without-fonts`!
6363
:::
6464

65-
If your font is a local file referenced via `@font-face`, it will be processed as an asset and included under `.vitepress/dist/assets` with hashed filename. To preload this file, use the [transformHead](/reference/site-config#transformhead) build hook:
65+
If your font is a local file referenced via `@font-face`, it will be processed as an asset and included under `.vitepress/dist/assets` with hashed filename. To preload this file, use the [transformHead](../reference/site-config#transformhead) build hook:
6666

6767
```js
6868
// .vitepress/config.js

docs/guide/ssr-compat.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -48,7 +48,7 @@ if (!import.meta.env.SSR) {
4848
}
4949
```
5050
51-
Since [`Theme.enhanceApp`](/guide/custom-theme#theme-interface) can be async, you can conditionally import and register Vue plugins that access browser APIs on import:
51+
Since [`Theme.enhanceApp`](./custom-theme#theme-interface) can be async, you can conditionally import and register Vue plugins that access browser APIs on import:
5252
5353
```js
5454
// .vitepress/theme/index.js

docs/reference/runtime-api.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -141,7 +141,7 @@ If you are using or demoing components that are not SSR-friendly (for example, c
141141
</ClientOnly>
142142
```
143143

144-
- Related: [SSR Compatibility](/guide/ssr-compat)
144+
- Related: [SSR Compatibility](../guide/ssr-compat)
145145

146146
## `$frontmatter` <Badge type="info" text="template global" />
147147

src/node/contentLoader.ts

+34-8
Original file line numberDiff line numberDiff line change
@@ -9,19 +9,41 @@ import { createMarkdownRenderer, type MarkdownRenderer } from './markdown'
99
export interface ContentOptions<T = ContentData[]> {
1010
/**
1111
* Include src?
12-
* default: false
12+
* @default false
1313
*/
1414
includeSrc?: boolean
15+
1516
/**
1617
* Render src to HTML and include in data?
17-
* default: false
18+
* @default false
1819
*/
1920
render?: boolean
21+
2022
/**
21-
* Whether to parse and include excerpt (rendered as HTML)
22-
* default: false
23+
* If `boolean`, whether to parse and include excerpt? (rendered as HTML)
24+
*
25+
* If `function`, control how the excerpt is extracted from the content.
26+
*
27+
* If `string`, define a custom separator to be used for extracting the
28+
* excerpt. Default separator is `---` if `excerpt` is `true`.
29+
*
30+
* @see https://github.com/jonschlinkert/gray-matter#optionsexcerpt
31+
* @see https://github.com/jonschlinkert/gray-matter#optionsexcerpt_separator
32+
*
33+
* @default false
2334
*/
24-
excerpt?: boolean
35+
excerpt?:
36+
| boolean
37+
| ((
38+
file: {
39+
data: { [key: string]: any }
40+
content: string
41+
excerpt?: string
42+
},
43+
options?: any
44+
) => void)
45+
| string
46+
2547
/**
2648
* Transform the data. Note the data will be inlined as JSON in the client
2749
* bundle if imported from components or markdown files.
@@ -110,9 +132,13 @@ export function createContentLoader<T = ContentData[]>(
110132
raw.push(cached.data)
111133
} else {
112134
const src = fs.readFileSync(file, 'utf-8')
113-
const { data: frontmatter, excerpt } = matter(src, {
114-
excerpt: true
115-
})
135+
const { data: frontmatter, excerpt } = matter(
136+
src,
137+
// @ts-expect-error gray-matter types are wrong
138+
typeof renderExcerpt === 'string'
139+
? { excerpt_separator: renderExcerpt }
140+
: { excerpt: renderExcerpt }
141+
)
116142
const url =
117143
'/' +
118144
normalizePath(path.relative(config.srcDir, file)).replace(

0 commit comments

Comments
 (0)