From de41860aaad7f55529fbc893f3cd1c7945acd411 Mon Sep 17 00:00:00 2001 From: Anton Bulakh Date: Tue, 24 Dec 2024 02:03:09 +0200 Subject: [PATCH] feat(frontmatter): Add an option to index frontmatter wikilinks MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit I was missing this feature and implemented it, not knowing that there was #944 already ¯\_(ツ)_/¯ Looking at it now I think that my impl is a little better, and I'd love to have it upstreamed. Closes #820 and #944 --- docs/plugins/CrawlLinks.md | 1 + quartz/plugins/transformers/links.ts | 53 ++++++++++++++++++++++------ 2 files changed, 43 insertions(+), 11 deletions(-) diff --git a/docs/plugins/CrawlLinks.md b/docs/plugins/CrawlLinks.md index 47b7bdd776133..5bc92a5c1f18e 100644 --- a/docs/plugins/CrawlLinks.md +++ b/docs/plugins/CrawlLinks.md @@ -19,6 +19,7 @@ This plugin accepts the following configuration options: - `openLinksInNewTab`: If `true`, configures external links to open in a new tab. Defaults to `false`. - `lazyLoad`: If `true`, adds lazy loading to resource elements (`img`, `video`, etc.) to improve page load performance. Defaults to `false`. - `externalLinkIcon`: Adds an icon next to external links when `true` (default) to visually distinguishing them from internal links. +- `indexFrontmatterWikilinks`: If `true`, parses Obsidian-style wikilinks in the frontmatter and adds them to the graph (including things like backlinks) as if they were part of the note content. Defaults to `false`. > [!warning] > Removing this plugin is _not_ recommended and will likely break the page. diff --git a/quartz/plugins/transformers/links.ts b/quartz/plugins/transformers/links.ts index 3e8dbdede5ba5..88d1b93d82c50 100644 --- a/quartz/plugins/transformers/links.ts +++ b/quartz/plugins/transformers/links.ts @@ -13,6 +13,7 @@ import path from "path" import { visit } from "unist-util-visit" import isAbsoluteUrl from "is-absolute-url" import { Root } from "hast" +import { wikilinkRegex } from "./ofm" interface Options { /** How to resolve Markdown paths */ @@ -22,6 +23,7 @@ interface Options { openLinksInNewTab: boolean lazyLoad: boolean externalLinkIcon: boolean + indexFrontmatterWikilinks: boolean } const defaultOptions: Options = { @@ -30,6 +32,20 @@ const defaultOptions: Options = { openLinksInNewTab: false, lazyLoad: false, externalLinkIcon: true, + indexFrontmatterWikilinks: false, +} + +function getFullInternalLink(dest: RelativeURL, fileSlug: SimpleSlug): FullSlug { + // url.resolve is considered legacy + // WHATWG equivalent https://nodejs.dev/en/api/v18/url/#urlresolvefrom-to + const url = new URL(dest, "https://base.com/" + stripSlashes(fileSlug, true)) + const canonicalDest = url.pathname + let [destCanonical, _destAnchor] = splitAnchor(canonicalDest) + if (destCanonical.endsWith("/")) { + destCanonical += "index" + } + // need to decodeURIComponent here as WHATWG URL percent-encodes everything + return decodeURIComponent(stripSlashes(destCanonical, true)) as FullSlug } export const CrawlLinks: QuartzTransformerPlugin> = (userOpts) => { @@ -107,17 +123,7 @@ export const CrawlLinks: QuartzTransformerPlugin> = (userOpts) transformOptions, ) - // url.resolve is considered legacy - // WHATWG equivalent https://nodejs.dev/en/api/v18/url/#urlresolvefrom-to - const url = new URL(dest, "https://base.com/" + stripSlashes(curSlug, true)) - const canonicalDest = url.pathname - let [destCanonical, _destAnchor] = splitAnchor(canonicalDest) - if (destCanonical.endsWith("/")) { - destCanonical += "index" - } - - // need to decodeURIComponent here as WHATWG URL percent-encodes everything - const full = decodeURIComponent(stripSlashes(destCanonical, true)) as FullSlug + const full = getFullInternalLink(dest, curSlug) const simple = simplifySlug(full) outgoing.add(simple) node.properties["data-slug"] = full @@ -157,6 +163,31 @@ export const CrawlLinks: QuartzTransformerPlugin> = (userOpts) } }) + if (opts.indexFrontmatterWikilinks) { + const strings = Object.values(file.data.frontmatter ?? {}) + .flatMap((vs) => (Array.isArray(vs) ? vs : [vs])) + .filter((v) => typeof v === "string") + + for (const string of strings) { + // the regex is /g so we have to do this to get the captures + // exec doesn't work because it's stateful and so returns null every other time (very bad) + // we do all of that to reuse the wikilinkRegex from ofm + const [captures] = [...string.matchAll(wikilinkRegex)] + if (!captures || captures[0] != string || string.startsWith("!")) { + // not matched, or didn't match the whole string, or is the embed syntax for some reason, + // which doesn't make sense to support in frontmatter + continue + } + const [_, rawFp, rawHeader] = captures + const fp = rawFp?.trim() ?? "" + const anchor = rawHeader?.trim() ?? "" + const dest = transformLink(file.data.slug!, fp + anchor, transformOptions) + const full = getFullInternalLink(dest, curSlug) + const simple = simplifySlug(full) + outgoing.add(simple) + } + } + file.data.links = [...outgoing] } },