inline scripts
This commit is contained in:
		| @@ -1,6 +1,7 @@ | ||||
| #!/usr/bin/env node | ||||
| import { readFileSync } from 'fs' | ||||
| import { promises, readFileSync } from 'fs' | ||||
| import yargs from 'yargs' | ||||
| import path from 'path' | ||||
| import { hideBin } from 'yargs/helpers' | ||||
| import esbuild from 'esbuild' | ||||
| import chalk from 'chalk' | ||||
| @@ -61,9 +62,34 @@ yargs(hideBin(process.argv)) | ||||
|       jsx: "automatic", | ||||
|       jsxImportSource: "preact", | ||||
|       external: ["@napi-rs/simple-git", "shiki"], | ||||
|       plugins: [sassPlugin({ | ||||
|         type: 'css-text' | ||||
|       })] | ||||
|       plugins: [ | ||||
|         sassPlugin({ | ||||
|           type: 'css-text' | ||||
|         }), | ||||
|         { | ||||
|           name: 'inline-script-loader', | ||||
|           setup(build) { | ||||
|             build.onLoad({ filter: /\.inline\.(ts|js)$/ }, async (args) => { | ||||
|               let text = await promises.readFile(args.path, 'utf8') | ||||
|               const transpiled = await esbuild.build({ | ||||
|                 stdin: { | ||||
|                   contents: text, | ||||
|                   sourcefile: path.relative(path.resolve('.'), args.path), | ||||
|                 }, | ||||
|                 write: false, | ||||
|                 bundle: true, | ||||
|                 platform: "browser", | ||||
|                 format: "esm", | ||||
|               }) | ||||
|               const rawMod = transpiled.outputFiles[0].text | ||||
|               return { | ||||
|                 contents: rawMod, | ||||
|                 loader: 'text', | ||||
|               } | ||||
|             }) | ||||
|           } | ||||
|         } | ||||
|       ] | ||||
|     }).catch(err => { | ||||
|       console.error(`${chalk.red("Couldn't parse Quartz configuration:")} ${fp}`) | ||||
|       console.log(`Reason: ${chalk.grey(err)}`) | ||||
|   | ||||
| @@ -8,10 +8,9 @@ export interface HeadProps { | ||||
|   externalResources: StaticResources | ||||
| } | ||||
|  | ||||
| export default function({ title, description, slug, externalResources }: HeadProps) { | ||||
| export function Component({ title, description, slug, externalResources }: HeadProps) { | ||||
|   const { css, js } = externalResources | ||||
|   const baseDir = resolveToRoot(slug) | ||||
|   const stylePath = baseDir + "/index.css" | ||||
|   const iconPath = baseDir + "/static/icon.png" | ||||
|   const ogImagePath = baseDir + "/static/og-image.png" | ||||
|   return <head> | ||||
| @@ -28,16 +27,7 @@ export default function({ title, description, slug, externalResources }: HeadPro | ||||
|     <meta name="generator" content="Quartz" /> | ||||
|     <link rel="preconnect" href="https://fonts.googleapis.com" /> | ||||
|     <link rel="preconnect" href="https://fonts.gstatic.com" /> | ||||
|     <link rel="stylesheet" type="text/css" href={stylePath} /> | ||||
|     {css.map(href => <link key={href} href={href} rel="stylesheet" type="text/css" />)} | ||||
|     {js.filter(resource => resource.loadTime === "beforeDOMReady").map(resource => <script key={resource.src} src={resource.src} />)} | ||||
|     {js.filter(resource => resource.loadTime === "beforeDOMReady").map(resource => <script key={resource.src} {...resource} />)} | ||||
|   </head> | ||||
| } | ||||
|  | ||||
| export function beforeDOMLoaded() { | ||||
|  | ||||
| } | ||||
|  | ||||
| export function onDOMLoaded() { | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -5,10 +5,10 @@ export interface HeaderProps { | ||||
|   slug: string | ||||
| } | ||||
|  | ||||
| export default function({ title, slug }: HeaderProps) { | ||||
| export function Component({ title, slug }: HeaderProps) { | ||||
|   const baseDir = resolveToRoot(slug) | ||||
|   return <header> | ||||
|     <h1><a href={baseDir}>{title}</a></h1> | ||||
|   </header> | ||||
|  | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										3
									
								
								quartz/components/scripts/darkmode.inline.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								quartz/components/scripts/darkmode.inline.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| export default "Darkmode"  | ||||
|  | ||||
| console.log("HELLOOOO FROM CONSOLE") | ||||
							
								
								
									
										8
									
								
								quartz/components/types.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								quartz/components/types.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| import { ComponentType } from "preact" | ||||
|  | ||||
| export type QuartzComponent<Props> = { | ||||
|   Component: ComponentType<Props> | ||||
|   css?: string, | ||||
|   beforeDOMLoaded?: string, | ||||
|   afterDOMLoaded?: string, | ||||
| } | ||||
| @@ -60,7 +60,7 @@ export function buildQuartz(cfg: QuartzConfig) { | ||||
|     const parsedFiles = await parseMarkdown(processor, argv.directory, filePaths, argv.verbose) | ||||
|     const filteredContent = filterContent(cfg.plugins.filters, parsedFiles, argv.verbose) | ||||
|     await emitContent(argv.directory, output, cfg, filteredContent, argv.verbose) | ||||
|     console.log(chalk.green(`Done in ${perf.timeSince()}`)) | ||||
|     console.log(chalk.green(`Done processing ${fps.length} files in ${perf.timeSince()}`)) | ||||
|  | ||||
|     if (argv.serve) { | ||||
|       const server = http.createServer(async (req, res) => { | ||||
|   | ||||
| @@ -22,11 +22,15 @@ export function slugify(s: string): string { | ||||
| // resolve /a/b/c to ../../ | ||||
| export function resolveToRoot(slug: string): string { | ||||
|   let fp = slug | ||||
|   if (fp.endsWith("/index")) { | ||||
|     fp = fp.slice(0, -"/index".length) | ||||
|   if (fp.endsWith("index")) { | ||||
|     fp = fp.slice(0, -"index".length) | ||||
|   } | ||||
|  | ||||
|   return fp | ||||
|   if (fp === "") { | ||||
|     return "." | ||||
|   } | ||||
|  | ||||
|   return "./" + fp | ||||
|     .split('/') | ||||
|     .filter(x => x !== '') | ||||
|     .map(_ => '..') | ||||
|   | ||||
| @@ -4,17 +4,15 @@ import { EmitCallback, QuartzEmitterPlugin } from "../types" | ||||
| import { ProcessedContent } from "../vfile" | ||||
| import { Fragment, jsx, jsxs } from 'preact/jsx-runtime' | ||||
| import { render } from "preact-render-to-string" | ||||
| import { ComponentType } from "preact" | ||||
| import { HeadProps } from "../../components/Head" | ||||
| import { googleFontHref, templateThemeStyles } from "../../theme" | ||||
| import { GlobalConfiguration } from "../../cfg" | ||||
| import { HeaderProps } from "../../components/Header" | ||||
|  | ||||
| import styles from '../../styles/base.scss' | ||||
| import { QuartzComponent } from "../../components/types" | ||||
| import { resolveToRoot } from "../../path" | ||||
|  | ||||
| interface Options { | ||||
|   Head: ComponentType<HeadProps> | ||||
|   Header: ComponentType<HeaderProps> | ||||
|   Head: QuartzComponent<HeadProps> | ||||
|   Header: QuartzComponent<HeaderProps> | ||||
| } | ||||
|  | ||||
| export class ContentPage extends QuartzEmitterPlugin { | ||||
| @@ -26,40 +24,45 @@ export class ContentPage extends QuartzEmitterPlugin { | ||||
|     this.opts = opts | ||||
|   } | ||||
|  | ||||
|   getQuartzComponents(): QuartzComponent<any>[] { | ||||
|     return [...Object.values(this.opts)] | ||||
|   } | ||||
|  | ||||
|   async emit(cfg: GlobalConfiguration, content: ProcessedContent[], resources: StaticResources, emit: EmitCallback): Promise<string[]> { | ||||
|     const fps: string[] = [] | ||||
|  | ||||
|     // emit styles | ||||
|     emit({ | ||||
|       slug: "index", | ||||
|       ext: ".css", | ||||
|       content: templateThemeStyles(cfg.theme, styles) | ||||
|     }) | ||||
|     fps.push("index.css") | ||||
|     resources.css.push(googleFontHref(cfg.theme)) | ||||
|  | ||||
|     const { Head, Header } = this.opts | ||||
|     for (const [tree, file] of content) { | ||||
|       // @ts-ignore (preact makes it angry) | ||||
|       const content = toJsxRuntime(tree, { Fragment, jsx, jsxs, elementAttributeNameCase: 'html' }) | ||||
|  | ||||
|       const baseDir = resolveToRoot(file.data.slug!) | ||||
|       const pageResources: StaticResources = { | ||||
|         css: [baseDir + "/index.css", ...resources.css,], | ||||
|         js: [ | ||||
|           { src: baseDir + "/prescript.js", loadTime: "beforeDOMReady", type: 'module' }, | ||||
|           ...resources.js, | ||||
|           { src: baseDir + "/postscript.js", loadTime: "afterDOMReady", type: 'module' } | ||||
|         ] | ||||
|       } | ||||
|  | ||||
|       const title = file.data.frontmatter?.title | ||||
|       const { Head, Header } = this.opts | ||||
|       const doc = <html> | ||||
|         <Head | ||||
|         <Head.Component | ||||
|           title={title ?? "Untitled"} | ||||
|           description={file.data.description ?? "No description provided"} | ||||
|           slug={file.data.slug!} | ||||
|           externalResources={resources} /> | ||||
|           externalResources={pageResources} /> | ||||
|         <body> | ||||
|           <div id="quartz-root" class="page"> | ||||
|             <Header title={cfg.siteTitle} slug={file.data.slug!} /> | ||||
|             <Header.Component title={cfg.siteTitle} slug={file.data.slug!} /> | ||||
|             <article> | ||||
|               {file.data.slug !== "index" && <h1>{title}</h1>} | ||||
|               {content} | ||||
|             </article> | ||||
|           </div> | ||||
|         </body> | ||||
|         {resources.js.filter(resource => resource.loadTime === "afterDOMReady").map(resource => <script key={resource.src} src={resource.src} />)} | ||||
|         {pageResources.js.filter(resource => resource.loadTime === "afterDOMReady").map(resource => <script key={resource.src} {...resource} />)} | ||||
|       </html> | ||||
|  | ||||
|       const fp = file.data.slug + ".html" | ||||
|   | ||||
| @@ -1,5 +1,69 @@ | ||||
| import { GlobalConfiguration } from '../cfg' | ||||
| import { QuartzComponent } from '../components/types' | ||||
| import { StaticResources } from '../resources' | ||||
| import { PluginTypes } from './types' | ||||
| import { googleFontHref, joinStyles } from '../theme' | ||||
| import { EmitCallback, PluginTypes } from './types' | ||||
| import styles from '../styles/base.scss' | ||||
|  | ||||
| export type ComponentResources = { | ||||
|   css: string[], | ||||
|   beforeDOMLoaded: string[], | ||||
|   afterDOMLoaded: string[] | ||||
| } | ||||
|  | ||||
| function joinScripts(scripts: string[]): string { | ||||
|   return scripts.join("\n") | ||||
| } | ||||
|  | ||||
| export function emitComponentResources(cfg: GlobalConfiguration, resources: StaticResources, plugins: PluginTypes, emit: EmitCallback) { | ||||
|   const fps: string[] = [] | ||||
|   const allComponents: Set<QuartzComponent<any>> = new Set() | ||||
|   for (const emitter of plugins.emitters) { | ||||
|     const components = emitter.getQuartzComponents() | ||||
|     for (const component of components) { | ||||
|       allComponents.add(component) | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   const componentResources: ComponentResources = { | ||||
|     css: [], | ||||
|     beforeDOMLoaded: [], | ||||
|     afterDOMLoaded: [] | ||||
|   } | ||||
|  | ||||
|   for (const component of allComponents) { | ||||
|     const { css, beforeDOMLoaded, afterDOMLoaded } = component | ||||
|     if (css) { | ||||
|       componentResources.css.push(css) | ||||
|     } | ||||
|     if (beforeDOMLoaded) { | ||||
|       componentResources.beforeDOMLoaded.push(beforeDOMLoaded) | ||||
|     } | ||||
|     if (afterDOMLoaded) { | ||||
|       componentResources.beforeDOMLoaded.push(afterDOMLoaded) | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   emit({ | ||||
|     slug: "index", | ||||
|     ext: ".css", | ||||
|     content: joinStyles(cfg.theme, styles, ...componentResources.css) | ||||
|   }) | ||||
|   emit({ | ||||
|     slug: "prescript", | ||||
|     ext: ".js", | ||||
|     content: joinScripts(componentResources.beforeDOMLoaded) | ||||
|   }) | ||||
|   emit({ | ||||
|     slug: "postscript", | ||||
|     ext: ".js", | ||||
|     content: joinScripts(componentResources.afterDOMLoaded) | ||||
|   }) | ||||
|  | ||||
|   fps.push("index.css", "prescript.js", "postscript.js") | ||||
|   resources.css.push(googleFontHref(cfg.theme)) | ||||
|   return fps | ||||
| } | ||||
|  | ||||
| export function getStaticResourcesFromPlugins(plugins: PluginTypes) { | ||||
|   const staticResources: StaticResources = { | ||||
| @@ -7,8 +71,8 @@ export function getStaticResourcesFromPlugins(plugins: PluginTypes) { | ||||
|     js: [], | ||||
|   } | ||||
|  | ||||
|   for (const plugin of plugins.transformers) { | ||||
|     const res = plugin.externalResources | ||||
|   for (const transformer of plugins.transformers) { | ||||
|     const res = transformer.externalResources | ||||
|     if (res?.js) { | ||||
|       staticResources.js = staticResources.js.concat(res.js) | ||||
|     } | ||||
|   | ||||
| @@ -12,7 +12,7 @@ export interface Options { | ||||
|  | ||||
| const defaultOptions: Options = { | ||||
|   highlight: true, | ||||
|   wikilinks: true | ||||
|   wikilinks: true, | ||||
| } | ||||
|  | ||||
| export class ObsidianFlavoredMarkdown extends QuartzTransformerPlugin { | ||||
| @@ -39,10 +39,10 @@ export class ObsidianFlavoredMarkdown extends QuartzTransformerPlugin { | ||||
|         return (tree: Root, _file) => { | ||||
|           findAndReplace(tree, backlinkRegex, (value: string, ...capture: string[]) => { | ||||
|             if (value.startsWith("!")) { | ||||
|  | ||||
|               // TODO: handle embeds | ||||
|             } else { | ||||
|               const [path, rawHeader, rawAlias] = capture | ||||
|               const anchor = rawHeader?.slice(1).trim() ?? "" | ||||
|               const anchor = rawHeader?.trim() ?? "" | ||||
|               const alias = rawAlias?.slice(1).trim() ?? path | ||||
|               const url = slugify(path.trim() + anchor) | ||||
|               return { | ||||
|   | ||||
| @@ -2,6 +2,7 @@ import { PluggableList } from "unified" | ||||
| import { StaticResources } from "../resources" | ||||
| import { ProcessedContent } from "./vfile" | ||||
| import { GlobalConfiguration } from "../cfg" | ||||
| import { QuartzComponent } from "../components/types" | ||||
|  | ||||
| export abstract class QuartzTransformerPlugin { | ||||
|   abstract name: string | ||||
| @@ -25,6 +26,7 @@ export type EmitCallback = (data: EmitOptions) => Promise<string> | ||||
| export abstract class QuartzEmitterPlugin { | ||||
|   abstract name: string | ||||
|   abstract emit(cfg: GlobalConfiguration, content: ProcessedContent[], resources: StaticResources, emitCallback: EmitCallback): Promise<string[]> | ||||
|   abstract getQuartzComponents(): QuartzComponent<any>[] | ||||
| } | ||||
|  | ||||
| export interface PluginTypes { | ||||
|   | ||||
| @@ -2,7 +2,7 @@ import path from "path" | ||||
| import fs from "fs" | ||||
| import { QuartzConfig } from "../cfg" | ||||
| import { PerfTimer } from "../perf" | ||||
| import { getStaticResourcesFromPlugins } from "../plugins" | ||||
| import { emitComponentResources, getStaticResourcesFromPlugins } from "../plugins" | ||||
| import { EmitCallback } from "../plugins/types" | ||||
| import { ProcessedContent } from "../plugins/vfile" | ||||
| import { QUARTZ, slugify } from "../path" | ||||
| @@ -10,9 +10,6 @@ import { globbyStream } from "globby" | ||||
|  | ||||
| export async function emitContent(contentFolder: string, output: string, cfg: QuartzConfig, content: ProcessedContent[], verbose: boolean) { | ||||
|   const perf = new PerfTimer() | ||||
|  | ||||
|  | ||||
|   const staticResources = getStaticResourcesFromPlugins(cfg.plugins) | ||||
|   const emit: EmitCallback = async ({ slug, ext, content }) => { | ||||
|     const pathToPage = path.join(output, slug + ext) | ||||
|     const dir = path.dirname(pathToPage) | ||||
| @@ -21,6 +18,9 @@ export async function emitContent(contentFolder: string, output: string, cfg: Qu | ||||
|     return pathToPage | ||||
|   } | ||||
|  | ||||
|   const staticResources = getStaticResourcesFromPlugins(cfg.plugins) | ||||
|   emitComponentResources(cfg.configuration, staticResources, cfg.plugins, emit) | ||||
|  | ||||
|   let emittedFiles = 0 | ||||
|   for (const emitter of cfg.plugins.emitters) { | ||||
|     const emitted = await emitter.emit(cfg.configuration, content, staticResources, emit) | ||||
| @@ -35,6 +35,9 @@ export async function emitContent(contentFolder: string, output: string, cfg: Qu | ||||
|  | ||||
|   const staticPath = path.join(QUARTZ, "static") | ||||
|   await fs.promises.cp(staticPath, path.join(output, "static"), { recursive: true }) | ||||
|   if (verbose) { | ||||
|     console.log(`[emit:Static] ${path.join(output, "static", "**")}`) | ||||
|   } | ||||
|  | ||||
|   // glob all non MD/MDX/HTML files in content folder and copy it over | ||||
|   const assetsPath = path.join("public", "assets") | ||||
| @@ -54,8 +57,5 @@ export async function emitContent(contentFolder: string, output: string, cfg: Qu | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (verbose) { | ||||
|     console.log(`[emit:Static] ${path.join(output, "static", "**")}`) | ||||
|     console.log(`Emitted ${emittedFiles} files to \`${output}\` in ${perf.timeSince()}`) | ||||
|   } | ||||
|   console.log(`Emitted ${emittedFiles} files to \`${output}\` in ${perf.timeSince()}`) | ||||
| } | ||||
|   | ||||
| @@ -6,11 +6,18 @@ export function filterContent(plugins: QuartzFilterPlugin[], content: ProcessedC | ||||
|   const perf = new PerfTimer() | ||||
|   const initialLength = content.length | ||||
|   for (const plugin of plugins) { | ||||
|     content = content.filter(plugin.shouldPublish) | ||||
|     const updatedContent = content.filter(plugin.shouldPublish) | ||||
|  | ||||
|     if (verbose) { | ||||
|       const diff = content.filter(x => !updatedContent.includes(x)) | ||||
|       for (const file of diff) { | ||||
|         console.log(`[filter:${plugin.name}] ${file[1].data.slug}`) | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     content = updatedContent | ||||
|   } | ||||
|  | ||||
|   if (verbose) { | ||||
|     console.log(`Filtered out ${initialLength - content.length} files in ${perf.timeSince()}`) | ||||
|   } | ||||
|   console.log(`Filtered out ${initialLength - content.length} files in ${perf.timeSince()}`) | ||||
|   return content | ||||
| } | ||||
|   | ||||
| @@ -50,9 +50,6 @@ export async function parseMarkdown(processor: QuartzProcessor, baseDir: string, | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (verbose) { | ||||
|     console.log(`Parsed and transformed ${res.length} Markdown files in ${perf.timeSince()}`) | ||||
|   } | ||||
|  | ||||
|   console.log(`Parsed and transformed ${res.length} Markdown files in ${perf.timeSince()}`) | ||||
|   return res | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| export interface JSResource { | ||||
|   src: string | ||||
|   loadTime: 'beforeDOMReady' | 'afterDOMReady' | ||||
|   type?: 'module' | ||||
| } | ||||
|  | ||||
| export interface StaticResources { | ||||
|   | ||||
| @@ -98,6 +98,8 @@ h1, h2, h3, h4, h5, h6 { | ||||
|     margin: 0 0.5rem; | ||||
|     opacity: 0; | ||||
|     transition: opacity 0.2s ease; | ||||
|     transform: translateY(-0.1rem); | ||||
|     display: inline-block; | ||||
|     font-family: var(--codeFont); | ||||
|     user-select: none; | ||||
|   } | ||||
|   | ||||
| @@ -26,9 +26,8 @@ export function googleFontHref(theme: Theme) { | ||||
|   return `https://fonts.googleapis.com/css2?family=${code}&family=${header}:wght@400;700&family=${body}:ital,wght@0,400;0,600;1,400;1,600&display=swap` | ||||
| } | ||||
|  | ||||
| export function templateThemeStyles(theme: Theme, stylesheet: string) { | ||||
|   return ` | ||||
| :root { | ||||
| export function joinStyles(theme: Theme, ...stylesheet: string[]) { | ||||
|   return `:root { | ||||
|   --light: ${theme.colors.lightMode.light}; | ||||
|   --lightgray: ${theme.colors.lightMode.lightgray}; | ||||
|   --gray: ${theme.colors.lightMode.gray}; | ||||
| @@ -54,6 +53,5 @@ export function templateThemeStyles(theme: Theme, stylesheet: string) { | ||||
|   --highlight: ${theme.colors.darkMode.highlight}; | ||||
| } | ||||
|  | ||||
| ${stylesheet} | ||||
| ` | ||||
| ${stylesheet.join("\n\n")}` | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user