| | |
| | | import { ProcessedContent, defaultProcessedContent } from "../vfile" |
| | | import { FullPageLayout } from "../../cfg" |
| | | import path from "path" |
| | | import { CanonicalSlug, FilePath, ServerSlug, canonicalizeServer, joinSegments } from "../../path" |
| | | import { FilePath, FullSlug, SimpleSlug, joinSegments, simplifySlug } from "../../util/path" |
| | | import { defaultListPageLayout, sharedPageComponents } from "../../../quartz.layout" |
| | | import { FolderContent } from "../../components" |
| | | |
| | | export const FolderPage: QuartzEmitterPlugin<FullPageLayout> = (opts) => { |
| | | if (!opts) { |
| | | throw new Error("ErrorPage must be initialized with options specifiying the components to use") |
| | | export const FolderPage: QuartzEmitterPlugin<FullPageLayout> = (userOpts) => { |
| | | const opts: FullPageLayout = { |
| | | ...sharedPageComponents, |
| | | ...defaultListPageLayout, |
| | | pageBody: FolderContent(), |
| | | ...userOpts, |
| | | } |
| | | |
| | | const { head: Head, header, beforeBody, pageBody: Content, left, right, footer: Footer } = opts |
| | | const { head: Head, header, beforeBody, pageBody, left, right, footer: Footer } = opts |
| | | const Header = HeaderConstructor() |
| | | const Body = BodyConstructor() |
| | | |
| | | return { |
| | | name: "FolderPage", |
| | | getQuartzComponents() { |
| | | return [Head, Header, Body, ...header, ...beforeBody, Content, ...left, ...right, Footer] |
| | | return [Head, Header, Body, ...header, ...beforeBody, pageBody, ...left, ...right, Footer] |
| | | }, |
| | | async emit(_contentDir, cfg, content, resources, emit): Promise<FilePath[]> { |
| | | async emit(ctx, content, resources, emit): Promise<FilePath[]> { |
| | | const fps: FilePath[] = [] |
| | | const allFiles = content.map(c => c[1].data) |
| | | const allFiles = content.map((c) => c[1].data) |
| | | const cfg = ctx.cfg.configuration |
| | | |
| | | const folders: Set<CanonicalSlug> = new Set(allFiles.flatMap(data => { |
| | | const slug = data.slug |
| | | const folderName = path.dirname(slug ?? "") as CanonicalSlug |
| | | if (slug && folderName !== ".") { |
| | | return [folderName] |
| | | } |
| | | return [] |
| | | })) |
| | | const folders: Set<SimpleSlug> = new Set( |
| | | allFiles.flatMap((data) => { |
| | | const slug = data.slug |
| | | const folderName = path.dirname(slug ?? "") as SimpleSlug |
| | | if (slug && folderName !== "." && folderName !== "tags") { |
| | | return [folderName] |
| | | } |
| | | return [] |
| | | }), |
| | | ) |
| | | |
| | | // remove special prefixes |
| | | folders.delete("tags" as CanonicalSlug) |
| | | |
| | | const folderDescriptions: Record<string, ProcessedContent> = Object.fromEntries([...folders].map(folder => ([ |
| | | folder, defaultProcessedContent({ slug: joinSegments(folder, "index") as ServerSlug, frontmatter: { title: `Folder: ${folder}`, tags: [] } }) |
| | | ]))) |
| | | const folderDescriptions: Record<string, ProcessedContent> = Object.fromEntries( |
| | | [...folders].map((folder) => [ |
| | | folder, |
| | | defaultProcessedContent({ |
| | | slug: joinSegments(folder, "index") as FullSlug, |
| | | frontmatter: { title: `Folder: ${folder}`, tags: [] }, |
| | | }), |
| | | ]), |
| | | ) |
| | | |
| | | for (const [tree, file] of content) { |
| | | const slug = canonicalizeServer(file.data.slug!) |
| | | const slug = simplifySlug(file.data.slug!) |
| | | if (folders.has(slug)) { |
| | | folderDescriptions[slug] = [tree, file] |
| | | } |
| | | } |
| | | |
| | | for (const folder of folders) { |
| | | const slug = folder |
| | | const slug = joinSegments(folder, "index") as FullSlug |
| | | const externalResources = pageResources(slug, resources) |
| | | const [tree, file] = folderDescriptions[folder] |
| | | const componentData: QuartzComponentProps = { |
| | |
| | | cfg, |
| | | children: [], |
| | | tree, |
| | | allFiles |
| | | allFiles, |
| | | } |
| | | |
| | | const content = renderPage( |
| | | slug, |
| | | componentData, |
| | | opts, |
| | | externalResources |
| | | ) |
| | | |
| | | const fp = file.data.slug! + ".html" as FilePath |
| | | await emit({ |
| | | const content = renderPage(slug, componentData, opts, externalResources) |
| | | const fp = await emit({ |
| | | content, |
| | | slug: file.data.slug!, |
| | | slug, |
| | | ext: ".html", |
| | | }) |
| | | |
| | | fps.push(fp) |
| | | } |
| | | return fps |
| | | } |
| | | }, |
| | | } |
| | | } |