2023-07-23 00:27:41 +00:00
|
|
|
import { render } from "preact-render-to-string"
|
|
|
|
import { QuartzComponent, QuartzComponentProps } from "./types"
|
2023-07-01 07:03:01 +00:00
|
|
|
import HeaderConstructor from "./Header"
|
|
|
|
import BodyConstructor from "./Body"
|
2023-08-17 05:04:15 +00:00
|
|
|
import { JSResourceToScriptElement, StaticResources } from "../util/resources"
|
2024-01-05 08:29:34 +00:00
|
|
|
import { FullSlug, RelativeURL, joinSegments, normalizeHastElement } from "../util/path"
|
2023-09-13 18:28:53 +00:00
|
|
|
import { visit } from "unist-util-visit"
|
2023-11-14 06:51:40 +00:00
|
|
|
import { Root, Element, ElementContent } from "hast"
|
2024-01-05 08:29:34 +00:00
|
|
|
import { QuartzPluginData } from "../plugins/vfile"
|
2023-07-01 07:03:01 +00:00
|
|
|
|
|
|
|
interface RenderComponents {
|
|
|
|
head: QuartzComponent
|
2023-07-23 00:27:41 +00:00
|
|
|
header: QuartzComponent[]
|
|
|
|
beforeBody: QuartzComponent[]
|
|
|
|
pageBody: QuartzComponent
|
|
|
|
left: QuartzComponent[]
|
|
|
|
right: QuartzComponent[]
|
|
|
|
footer: QuartzComponent
|
2023-07-01 07:03:01 +00:00
|
|
|
}
|
|
|
|
|
2023-09-13 04:29:57 +00:00
|
|
|
export function pageResources(
|
|
|
|
baseDir: FullSlug | RelativeURL,
|
|
|
|
staticResources: StaticResources,
|
|
|
|
): StaticResources {
|
2023-08-19 22:52:25 +00:00
|
|
|
const contentIndexPath = joinSegments(baseDir, "static/contentIndex.json")
|
2024-01-05 08:29:34 +00:00
|
|
|
const contentIndexScript = `const fetchData = fetch("${contentIndexPath}").then(data => data.json())`
|
2023-07-02 20:08:29 +00:00
|
|
|
|
2023-07-01 07:03:01 +00:00
|
|
|
return {
|
2023-08-19 22:52:25 +00:00
|
|
|
css: [joinSegments(baseDir, "index.css"), ...staticResources.css],
|
2023-07-01 07:03:01 +00:00
|
|
|
js: [
|
2023-08-19 22:52:25 +00:00
|
|
|
{
|
2023-08-19 23:28:44 +00:00
|
|
|
src: joinSegments(baseDir, "prescript.js"),
|
2023-08-19 22:52:25 +00:00
|
|
|
loadTime: "beforeDOMReady",
|
|
|
|
contentType: "external",
|
|
|
|
},
|
2023-07-23 00:27:41 +00:00
|
|
|
{
|
|
|
|
loadTime: "beforeDOMReady",
|
|
|
|
contentType: "inline",
|
|
|
|
spaPreserve: true,
|
|
|
|
script: contentIndexScript,
|
|
|
|
},
|
2023-07-01 07:03:01 +00:00
|
|
|
...staticResources.js,
|
2023-07-23 00:27:41 +00:00
|
|
|
{
|
2023-08-19 23:28:44 +00:00
|
|
|
src: joinSegments(baseDir, "postscript.js"),
|
2023-07-23 00:27:41 +00:00
|
|
|
loadTime: "afterDOMReady",
|
|
|
|
moduleType: "module",
|
|
|
|
contentType: "external",
|
|
|
|
},
|
|
|
|
],
|
2023-07-01 07:03:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-05 08:29:34 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2023-07-23 00:27:41 +00:00
|
|
|
export function renderPage(
|
2023-08-19 22:52:25 +00:00
|
|
|
slug: FullSlug,
|
2023-07-23 00:27:41 +00:00
|
|
|
componentData: QuartzComponentProps,
|
|
|
|
components: RenderComponents,
|
|
|
|
pageResources: StaticResources,
|
|
|
|
): string {
|
2023-09-13 18:28:53 +00:00
|
|
|
// process transcludes in componentData
|
|
|
|
visit(componentData.tree as Root, "element", (node, _index, _parent) => {
|
|
|
|
if (node.tagName === "blockquote") {
|
|
|
|
const classNames = (node.properties?.className ?? []) as string[]
|
|
|
|
if (classNames.includes("transclude")) {
|
|
|
|
const inner = node.children[0] as Element
|
2024-01-05 08:29:34 +00:00
|
|
|
const transcludeTarget = inner.properties["data-slug"] as FullSlug
|
|
|
|
const page = getOrComputeFileIndex(componentData.allFiles).get(transcludeTarget)
|
2023-11-14 06:51:40 +00:00
|
|
|
if (!page) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-01-05 08:29:34 +00:00
|
|
|
let blockRef = node.properties.dataBlock as string | undefined
|
|
|
|
if (blockRef?.startsWith("#^")) {
|
2023-11-14 06:51:40 +00:00
|
|
|
// block transclude
|
2024-01-05 08:29:34 +00:00
|
|
|
blockRef = blockRef.slice("#^".length)
|
2023-11-14 06:51:40 +00:00
|
|
|
let blockNode = page.blocks?.[blockRef]
|
|
|
|
if (blockNode) {
|
|
|
|
if (blockNode.tagName === "li") {
|
|
|
|
blockNode = {
|
|
|
|
type: "element",
|
|
|
|
tagName: "ul",
|
2024-01-05 08:29:34 +00:00
|
|
|
properties: {},
|
2023-11-14 06:51:40 +00:00
|
|
|
children: [blockNode],
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
node.children = [
|
2024-01-05 08:29:34 +00:00
|
|
|
normalizeHastElement(blockNode, slug, transcludeTarget),
|
2023-11-14 06:51:40 +00:00
|
|
|
{
|
|
|
|
type: "element",
|
|
|
|
tagName: "a",
|
|
|
|
properties: { href: inner.properties?.href, class: ["internal"] },
|
|
|
|
children: [{ type: "text", value: `Link to original` }],
|
|
|
|
},
|
|
|
|
]
|
|
|
|
}
|
|
|
|
} else if (blockRef?.startsWith("#") && page.htmlAst) {
|
|
|
|
// header transclude
|
|
|
|
blockRef = blockRef.slice(1)
|
|
|
|
let startIdx = undefined
|
|
|
|
let endIdx = undefined
|
|
|
|
for (const [i, el] of page.htmlAst.children.entries()) {
|
|
|
|
if (el.type === "element" && el.tagName.match(/h[1-6]/)) {
|
|
|
|
if (endIdx) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2024-01-05 08:29:34 +00:00
|
|
|
if (startIdx !== undefined) {
|
2023-11-14 06:51:40 +00:00
|
|
|
endIdx = i
|
|
|
|
} else if (el.properties?.id === blockRef) {
|
|
|
|
startIdx = i
|
|
|
|
}
|
2023-09-13 18:28:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-05 08:29:34 +00:00
|
|
|
if (startIdx === undefined) {
|
2023-11-14 06:51:40 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
node.children = [
|
2024-01-05 08:29:34 +00:00
|
|
|
...(page.htmlAst.children.slice(startIdx, endIdx) as ElementContent[]).map((child) =>
|
|
|
|
normalizeHastElement(child as Element, slug, transcludeTarget),
|
|
|
|
),
|
2023-11-14 06:51:40 +00:00
|
|
|
{
|
|
|
|
type: "element",
|
|
|
|
tagName: "a",
|
|
|
|
properties: { href: inner.properties?.href, class: ["internal"] },
|
|
|
|
children: [{ type: "text", value: `Link to original` }],
|
|
|
|
},
|
|
|
|
]
|
|
|
|
} else if (page.htmlAst) {
|
|
|
|
// page transclude
|
2023-09-13 18:28:53 +00:00
|
|
|
node.children = [
|
2023-11-14 06:51:40 +00:00
|
|
|
{
|
|
|
|
type: "element",
|
|
|
|
tagName: "h1",
|
2024-01-05 08:29:34 +00:00
|
|
|
properties: {},
|
2023-11-14 06:51:40 +00:00
|
|
|
children: [
|
|
|
|
{ type: "text", value: page.frontmatter?.title ?? `Transclude of ${page.slug}` },
|
|
|
|
],
|
|
|
|
},
|
2024-01-05 08:29:34 +00:00
|
|
|
...(page.htmlAst.children as ElementContent[]).map((child) =>
|
|
|
|
normalizeHastElement(child as Element, slug, transcludeTarget),
|
|
|
|
),
|
2023-09-13 18:28:53 +00:00
|
|
|
{
|
|
|
|
type: "element",
|
|
|
|
tagName: "a",
|
|
|
|
properties: { href: inner.properties?.href, class: ["internal"] },
|
|
|
|
children: [{ type: "text", value: `Link to original` }],
|
|
|
|
},
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2023-07-23 00:27:41 +00:00
|
|
|
const {
|
|
|
|
head: Head,
|
|
|
|
header,
|
|
|
|
beforeBody,
|
|
|
|
pageBody: Content,
|
|
|
|
left,
|
|
|
|
right,
|
|
|
|
footer: Footer,
|
|
|
|
} = components
|
2023-07-01 07:03:01 +00:00
|
|
|
const Header = HeaderConstructor()
|
|
|
|
const Body = BodyConstructor()
|
|
|
|
|
2023-07-23 00:27:41 +00:00
|
|
|
const LeftComponent = (
|
2023-07-05 00:14:15 +00:00
|
|
|
<div class="left sidebar">
|
2023-07-23 00:27:41 +00:00
|
|
|
{left.map((BodyComponent) => (
|
|
|
|
<BodyComponent {...componentData} />
|
|
|
|
))}
|
2023-07-02 20:08:29 +00:00
|
|
|
</div>
|
2023-07-23 00:27:41 +00:00
|
|
|
)
|
2023-07-02 20:08:29 +00:00
|
|
|
|
2023-07-23 00:27:41 +00:00
|
|
|
const RightComponent = (
|
2023-07-05 00:14:15 +00:00
|
|
|
<div class="right sidebar">
|
2023-07-23 00:27:41 +00:00
|
|
|
{right.map((BodyComponent) => (
|
|
|
|
<BodyComponent {...componentData} />
|
|
|
|
))}
|
2023-07-02 20:08:29 +00:00
|
|
|
</div>
|
2023-07-23 00:27:41 +00:00
|
|
|
)
|
2023-07-02 20:08:29 +00:00
|
|
|
|
2023-07-23 00:27:41 +00:00
|
|
|
const doc = (
|
|
|
|
<html>
|
|
|
|
<Head {...componentData} />
|
|
|
|
<body data-slug={slug}>
|
|
|
|
<div id="quartz-root" class="page">
|
|
|
|
<Body {...componentData}>
|
|
|
|
{LeftComponent}
|
|
|
|
<div class="center">
|
|
|
|
<div class="page-header">
|
|
|
|
<Header {...componentData}>
|
|
|
|
{header.map((HeaderComponent) => (
|
|
|
|
<HeaderComponent {...componentData} />
|
|
|
|
))}
|
|
|
|
</Header>
|
|
|
|
<div class="popover-hint">
|
|
|
|
{beforeBody.map((BodyComponent) => (
|
|
|
|
<BodyComponent {...componentData} />
|
|
|
|
))}
|
|
|
|
</div>
|
2023-07-04 23:48:36 +00:00
|
|
|
</div>
|
2023-07-23 00:27:41 +00:00
|
|
|
<Content {...componentData} />
|
2023-07-04 23:48:36 +00:00
|
|
|
</div>
|
2023-07-23 00:27:41 +00:00
|
|
|
{RightComponent}
|
|
|
|
</Body>
|
|
|
|
<Footer {...componentData} />
|
|
|
|
</div>
|
|
|
|
</body>
|
|
|
|
{pageResources.js
|
|
|
|
.filter((resource) => resource.loadTime === "afterDOMReady")
|
|
|
|
.map((res) => JSResourceToScriptElement(res))}
|
|
|
|
</html>
|
|
|
|
)
|
2023-07-01 07:03:01 +00:00
|
|
|
|
|
|
|
return "<!DOCTYPE html>\n" + render(doc)
|
|
|
|
}
|