fix/feat(fast rebuild): re-render transclusions in normal and fastRebuild mode (#842)
* Re-render transclusions in normal watch mode * Include transclusions in ContentPage getDependencyGraph * Address PR comments
This commit is contained in:
parent
823d952922
commit
5af707ea20
2 changed files with 56 additions and 19 deletions
|
@ -3,10 +3,9 @@ import { QuartzComponent, QuartzComponentProps } from "./types"
|
||||||
import HeaderConstructor from "./Header"
|
import HeaderConstructor from "./Header"
|
||||||
import BodyConstructor from "./Body"
|
import BodyConstructor from "./Body"
|
||||||
import { JSResourceToScriptElement, StaticResources } from "../util/resources"
|
import { JSResourceToScriptElement, StaticResources } from "../util/resources"
|
||||||
import { FullSlug, RelativeURL, joinSegments, normalizeHastElement } from "../util/path"
|
import { clone, FullSlug, RelativeURL, joinSegments, normalizeHastElement } from "../util/path"
|
||||||
import { visit } from "unist-util-visit"
|
import { visit } from "unist-util-visit"
|
||||||
import { Root, Element, ElementContent } from "hast"
|
import { Root, Element, ElementContent } from "hast"
|
||||||
import { QuartzPluginData } from "../plugins/vfile"
|
|
||||||
import { GlobalConfiguration } from "../cfg"
|
import { GlobalConfiguration } from "../cfg"
|
||||||
import { i18n } from "../i18n"
|
import { i18n } from "../i18n"
|
||||||
|
|
||||||
|
@ -52,18 +51,6 @@ export function pageResources(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let pageIndex: Map<FullSlug, QuartzPluginData> | undefined = undefined
|
|
||||||
function getOrComputeFileIndex(allFiles: QuartzPluginData[]): Map<FullSlug, QuartzPluginData> {
|
|
||||||
if (!pageIndex) {
|
|
||||||
pageIndex = new Map()
|
|
||||||
for (const file of allFiles) {
|
|
||||||
pageIndex.set(file.slug!, file)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return pageIndex
|
|
||||||
}
|
|
||||||
|
|
||||||
export function renderPage(
|
export function renderPage(
|
||||||
cfg: GlobalConfiguration,
|
cfg: GlobalConfiguration,
|
||||||
slug: FullSlug,
|
slug: FullSlug,
|
||||||
|
@ -71,14 +58,18 @@ export function renderPage(
|
||||||
components: RenderComponents,
|
components: RenderComponents,
|
||||||
pageResources: StaticResources,
|
pageResources: StaticResources,
|
||||||
): string {
|
): string {
|
||||||
|
// make a deep copy of the tree so we don't remove the transclusion references
|
||||||
|
// for the file cached in contentMap in build.ts
|
||||||
|
const root = clone(componentData.tree) as Root
|
||||||
|
|
||||||
// process transcludes in componentData
|
// process transcludes in componentData
|
||||||
visit(componentData.tree as Root, "element", (node, _index, _parent) => {
|
visit(root, "element", (node, _index, _parent) => {
|
||||||
if (node.tagName === "blockquote") {
|
if (node.tagName === "blockquote") {
|
||||||
const classNames = (node.properties?.className ?? []) as string[]
|
const classNames = (node.properties?.className ?? []) as string[]
|
||||||
if (classNames.includes("transclude")) {
|
if (classNames.includes("transclude")) {
|
||||||
const inner = node.children[0] as Element
|
const inner = node.children[0] as Element
|
||||||
const transcludeTarget = inner.properties["data-slug"] as FullSlug
|
const transcludeTarget = inner.properties["data-slug"] as FullSlug
|
||||||
const page = getOrComputeFileIndex(componentData.allFiles).get(transcludeTarget)
|
const page = componentData.allFiles.find((f) => f.slug === transcludeTarget)
|
||||||
if (!page) {
|
if (!page) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -181,6 +172,9 @@ export function renderPage(
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// set componentData.tree to the edited html that has transclusions rendered
|
||||||
|
componentData.tree = root
|
||||||
|
|
||||||
const {
|
const {
|
||||||
head: Head,
|
head: Head,
|
||||||
header,
|
header,
|
||||||
|
|
|
@ -1,16 +1,56 @@
|
||||||
|
import path from "path"
|
||||||
|
import { visit } from "unist-util-visit"
|
||||||
|
import { Root } from "hast"
|
||||||
|
import { VFile } from "vfile"
|
||||||
import { QuartzEmitterPlugin } from "../types"
|
import { QuartzEmitterPlugin } from "../types"
|
||||||
import { QuartzComponentProps } from "../../components/types"
|
import { QuartzComponentProps } from "../../components/types"
|
||||||
import HeaderConstructor from "../../components/Header"
|
import HeaderConstructor from "../../components/Header"
|
||||||
import BodyConstructor from "../../components/Body"
|
import BodyConstructor from "../../components/Body"
|
||||||
import { pageResources, renderPage } from "../../components/renderPage"
|
import { pageResources, renderPage } from "../../components/renderPage"
|
||||||
import { FullPageLayout } from "../../cfg"
|
import { FullPageLayout } from "../../cfg"
|
||||||
import { FilePath, joinSegments, pathToRoot } from "../../util/path"
|
import { Argv } from "../../util/ctx"
|
||||||
|
import { FilePath, isRelativeURL, joinSegments, pathToRoot } from "../../util/path"
|
||||||
import { defaultContentPageLayout, sharedPageComponents } from "../../../quartz.layout"
|
import { defaultContentPageLayout, sharedPageComponents } from "../../../quartz.layout"
|
||||||
import { Content } from "../../components"
|
import { Content } from "../../components"
|
||||||
import chalk from "chalk"
|
import chalk from "chalk"
|
||||||
import { write } from "./helpers"
|
import { write } from "./helpers"
|
||||||
import DepGraph from "../../depgraph"
|
import DepGraph from "../../depgraph"
|
||||||
|
|
||||||
|
// get all the dependencies for the markdown file
|
||||||
|
// eg. images, scripts, stylesheets, transclusions
|
||||||
|
const parseDependencies = (argv: Argv, hast: Root, file: VFile): string[] => {
|
||||||
|
const dependencies: string[] = []
|
||||||
|
|
||||||
|
visit(hast, "element", (elem): void => {
|
||||||
|
let ref: string | null = null
|
||||||
|
|
||||||
|
if (
|
||||||
|
["script", "img", "audio", "video", "source", "iframe"].includes(elem.tagName) &&
|
||||||
|
elem?.properties?.src
|
||||||
|
) {
|
||||||
|
ref = elem.properties.src.toString()
|
||||||
|
} else if (["a", "link"].includes(elem.tagName) && elem?.properties?.href) {
|
||||||
|
// transclusions will create a tags with relative hrefs
|
||||||
|
ref = elem.properties.href.toString()
|
||||||
|
}
|
||||||
|
|
||||||
|
// if it is a relative url, its a local file and we need to add
|
||||||
|
// it to the dependency graph. otherwise, ignore
|
||||||
|
if (ref === null || !isRelativeURL(ref)) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
let fp = path.join(file.data.filePath!, path.relative(argv.directory, ref)).replace(/\\/g, "/")
|
||||||
|
// markdown files have the .md extension stripped in hrefs, add it back here
|
||||||
|
if (!fp.split("/").pop()?.includes(".")) {
|
||||||
|
fp += ".md"
|
||||||
|
}
|
||||||
|
dependencies.push(fp)
|
||||||
|
})
|
||||||
|
|
||||||
|
return dependencies
|
||||||
|
}
|
||||||
|
|
||||||
export const ContentPage: QuartzEmitterPlugin<Partial<FullPageLayout>> = (userOpts) => {
|
export const ContentPage: QuartzEmitterPlugin<Partial<FullPageLayout>> = (userOpts) => {
|
||||||
const opts: FullPageLayout = {
|
const opts: FullPageLayout = {
|
||||||
...sharedPageComponents,
|
...sharedPageComponents,
|
||||||
|
@ -29,13 +69,16 @@ export const ContentPage: QuartzEmitterPlugin<Partial<FullPageLayout>> = (userOp
|
||||||
return [Head, Header, Body, ...header, ...beforeBody, pageBody, ...left, ...right, Footer]
|
return [Head, Header, Body, ...header, ...beforeBody, pageBody, ...left, ...right, Footer]
|
||||||
},
|
},
|
||||||
async getDependencyGraph(ctx, content, _resources) {
|
async getDependencyGraph(ctx, content, _resources) {
|
||||||
// TODO handle transclusions
|
|
||||||
const graph = new DepGraph<FilePath>()
|
const graph = new DepGraph<FilePath>()
|
||||||
|
|
||||||
for (const [_tree, file] of content) {
|
for (const [tree, file] of content) {
|
||||||
const sourcePath = file.data.filePath!
|
const sourcePath = file.data.filePath!
|
||||||
const slug = file.data.slug!
|
const slug = file.data.slug!
|
||||||
graph.addEdge(sourcePath, joinSegments(ctx.argv.output, slug + ".html") as FilePath)
|
graph.addEdge(sourcePath, joinSegments(ctx.argv.output, slug + ".html") as FilePath)
|
||||||
|
|
||||||
|
parseDependencies(ctx.argv, tree as Root, file).forEach((dep) => {
|
||||||
|
graph.addEdge(dep as FilePath, sourcePath)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
return graph
|
return graph
|
||||||
|
|
Loading…
Reference in a new issue