jpg-quartz/quartz/plugins/transformers/frontmatter.ts

97 lines
2.8 KiB
TypeScript
Raw Normal View History

2023-05-30 15:02:20 +00:00
import matter from "gray-matter"
2023-07-23 00:27:41 +00:00
import remarkFrontmatter from "remark-frontmatter"
2023-05-30 15:02:20 +00:00
import { QuartzTransformerPlugin } from "../types"
2023-07-23 00:27:41 +00:00
import yaml from "js-yaml"
import toml from "toml"
import { slugTag } from "../../util/path"
2023-12-10 14:19:29 +00:00
import { QuartzPluginData } from "../vfile"
2023-05-30 15:02:20 +00:00
export interface Options {
delims: string | string[]
language: "yaml" | "toml"
2023-05-30 15:02:20 +00:00
}
const defaultOptions: Options = {
2023-07-23 00:27:41 +00:00
delims: "---",
language: "yaml",
}
function coalesceAliases(data: { [key: string]: any }, aliases: string[]) {
for (const alias of aliases) {
if (data[alias] !== undefined && data[alias] !== null) return data[alias]
}
}
function coerceToArray(input: string | string[]): string[] | undefined {
if (input === undefined || input === null) return undefined
// coerce to array
if (!Array.isArray(input)) {
input = input
.toString()
.split(",")
.map((tag: string) => tag.trim())
}
// remove all non-strings
return input
.filter((tag: unknown) => typeof tag === "string" || typeof tag === "number")
.map((tag: string | number) => tag.toString())
2023-05-30 15:02:20 +00:00
}
export const FrontMatter: QuartzTransformerPlugin<Partial<Options> | undefined> = (userOpts) => {
const opts = { ...defaultOptions, ...userOpts }
return {
name: "FrontMatter",
markdownPlugins() {
return [
[remarkFrontmatter, ["yaml", "toml"]],
() => {
return (_, file) => {
const { data } = matter(Buffer.from(file.value), {
2023-07-07 01:45:38 +00:00
...opts,
engines: {
2023-07-23 00:27:41 +00:00
yaml: (s) => yaml.load(s, { schema: yaml.JSON_SCHEMA }) as object,
toml: (s) => toml.parse(s) as object,
2023-07-23 00:27:41 +00:00
},
2023-07-07 01:45:38 +00:00
})
2023-12-10 14:19:29 +00:00
if (data.title) {
2023-09-26 01:15:55 +00:00
data.title = data.title.toString()
2023-12-10 14:19:29 +00:00
} else if (data.title === null || data.title === undefined) {
data.title = file.stem ?? "Untitled"
2023-09-26 01:15:55 +00:00
}
const tags = coerceToArray(coalesceAliases(data, ["tags", "tag"]))
if (tags) data.tags = [...new Set(tags.map((tag: string) => slugTag(tag)))]
const aliases = coerceToArray(coalesceAliases(data, ["aliases", "alias"]))
if (aliases) data.aliases = aliases
const cssclasses = coerceToArray(coalesceAliases(data, ["cssclasses", "cssclass"]))
if (cssclasses) data.cssclasses = cssclasses
2023-07-10 02:32:24 +00:00
// fill in frontmatter
2023-12-10 14:19:29 +00:00
file.data.frontmatter = data as QuartzPluginData["frontmatter"]
2023-05-30 15:02:20 +00:00
}
2023-07-23 00:27:41 +00:00
},
]
},
2023-05-30 15:02:20 +00:00
}
}
2023-07-23 00:27:41 +00:00
declare module "vfile" {
2023-05-30 15:02:20 +00:00
interface DataMap {
frontmatter: { [key: string]: unknown } & {
2023-05-30 15:02:20 +00:00
title: string
} & Partial<{
tags: string[]
aliases: string[]
description: string
publish: boolean
draft: boolean
enableToc: string
cssclasses: string[]
}>
2023-05-30 15:02:20 +00:00
}
}