diff --git a/docs/custom/config-parser.md b/docs/custom/config-parser.md index 0725bd3761..5dba75e88a 100644 --- a/docs/custom/config-parser.md +++ b/docs/custom/config-parser.md @@ -51,6 +51,7 @@ This example systematically replaces any `@@@` line with a line with `hello`. It - An extension can contain: - a `transformRawLines(lines)` function that runs just after parsing the headmatter of the md file and receives a list of all lines (from the md file). The function can mutate the list arbitrarily. - a `transformSlide(content, frontmatter)` function that is called for each slide, just after splitting the file, and receives the slide content as a string and the frontmatter of the slide as an object. The function can mutate the frontmatter and must return the content string (possibly modified, possibly `undefined` if no modifications have been done). + - a `transformNote(note, frontmatter)` function that is called for each slide, just after splitting the file, and receives the slide note as a string or undefined and the frontmatter of the slide as an object. The function can mutate the frontmatter and must return the note string (possibly modified, possibly `undefined` if no modifications have been done). - a `name` ## Example Preparser Extensions @@ -179,3 +180,57 @@ export default definePreparserSetup(() => { ``` And that's it. + + + +### Use case 3: using custom frontmatter to transform note + +Imagine a case where you want to replace the slides default notes with custom notes. +For instance, you might want to write your `slides.md` as follows: + + + +```md +--- +layout: quote +_note: notes/note.md +--- + +# Welcome + +> great! + + +``` + +Here we used an underscore in `_note` to avoid possible conflicts with existing frontmatter properties. + +To handle this `_note: ...` syntax in the frontmatter, create a `./setup/preparser.ts` file with the following content: + +```ts twoslash +import { definePreparserSetup } from '@slidev/types' +import fs from 'fs' +import { promises as fsp } from 'fs' + +export default definePreparserSetup(() => { + return [ + { + async transformNote(note, frontmatter) { + if ('_note' in frontmatter && fs.existsSync(frontmatter._note)) { + try { + const newNote = await fsp.readFile(frontmatter._note, 'utf8') + return newNote + } catch (err) { + } + } + + return note + }, + }, + ] +}) +``` + +And that's it. diff --git a/packages/parser/src/core.ts b/packages/parser/src/core.ts index 9e9656c4e2..2285222c1f 100644 --- a/packages/parser/src/core.ts +++ b/packages/parser/src/core.ts @@ -155,6 +155,12 @@ export async function parse( slide.level = slide.frontmatter.level } } + + if (e.transformNote) { + const newNote = await e.transformNote(slide.note, slide.frontmatter) + if (newNote !== undefined) + slide.note = newNote + } } } slides.push(slide) diff --git a/packages/types/src/types.ts b/packages/types/src/types.ts index ea96767431..988a68e83b 100644 --- a/packages/types/src/types.ts +++ b/packages/types/src/types.ts @@ -122,6 +122,7 @@ export interface SlidevPreparserExtension { name?: string transformRawLines?: (lines: string[]) => Promise | void transformSlide?: (content: string, frontmatter: any) => Promise + transformNote?: (note: string | undefined, frontmatter: any) => Promise } export type PreparserExtensionLoader = (headmatter: Record, filepath: string, mode?: string) => Promise