feat(fonts): fetch before build (#817)
* feat: fetch google fonts before build Signed-off-by: Aaron <29749331+aarnphm@users.noreply.github.com> * Update quartz/plugins/emitters/componentResources.ts * fix: fetching wolff2 Signed-off-by: Aaron <29749331+aarnphm@users.noreply.github.com> * chore: remove request stylesheet Signed-off-by: Aaron <29749331+aarnphm@users.noreply.github.com> * fix: race condition Signed-off-by: Aaron <29749331+aarnphm@users.noreply.github.com> * chore: remove preconnect for static fonts since we are already downloading fonts into public folder Signed-off-by: Aaron <29749331+aarnphm@users.noreply.github.com> * chore: remove deadcode Signed-off-by: Aaron <29749331+aarnphm@users.noreply.github.com> * chore: add options to gate for cdn caching Signed-off-by: Aaron <29749331+aarnphm@users.noreply.github.com> * Apply suggestions from code review Co-authored-by: Jacky Zhao <j.zhao2k19@gmail.com> * chore: apply jacky's suggestion Co-authored-by: Jacky Zhao <j.zhao2k19@gmail.com> * chore: add docs and only use one promise Signed-off-by: Aaron <29749331+aarnphm@users.noreply.github.com> * fix: fmt Signed-off-by: Aaron <29749331+aarnphm@users.noreply.github.com> * chore: remove deadcode * chore: final touches Co-authored-by: Jacky Zhao <j.zhao2k19@gmail.com> * revert: changes in theme.ts * fix: styles and remove deadcode Signed-off-by: Aaron <29749331+aarnphm@users.noreply.github.com> --------- Signed-off-by: Aaron <29749331+aarnphm@users.noreply.github.com> Co-authored-by: Jacky Zhao <j.zhao2k19@gmail.com>
This commit is contained in:
		| @@ -1,4 +1,4 @@ | ||||
| import { FilePath, FullSlug } from "../../util/path" | ||||
| import { FilePath, FullSlug, joinSegments } from "../../util/path" | ||||
| import { QuartzEmitterPlugin } from "../types" | ||||
|  | ||||
| // @ts-ignore | ||||
| @@ -172,27 +172,72 @@ export const ComponentResources: QuartzEmitterPlugin<Options> = (opts?: Partial< | ||||
|       return [] | ||||
|     }, | ||||
|     async emit(ctx, _content, resources): Promise<FilePath[]> { | ||||
|       const promises: Promise<FilePath>[] = [] | ||||
|       const cfg = ctx.cfg.configuration | ||||
|       // component specific scripts and styles | ||||
|       const componentResources = getComponentResources(ctx) | ||||
|       // important that this goes *after* component scripts | ||||
|       // as the "nav" event gets triggered here and we should make sure | ||||
|       // that everyone else had the chance to register a listener for it | ||||
|  | ||||
|       if (fontOrigin === "googleFonts") { | ||||
|         resources.css.push(googleFontHref(ctx.cfg.configuration.theme)) | ||||
|       } else if (fontOrigin === "local") { | ||||
|       let googleFontsStyleSheet = "" | ||||
|       if (fontOrigin === "local") { | ||||
|         // let the user do it themselves in css | ||||
|       } else if (fontOrigin === "googleFonts") { | ||||
|         if (cfg.theme.cdnCaching) { | ||||
|           resources.css.push(googleFontHref(cfg.theme)) | ||||
|         } else { | ||||
|           let match | ||||
|  | ||||
|           const fontSourceRegex = /url\((https:\/\/fonts.gstatic.com\/s\/[^)]+\.(woff2|ttf))\)/g | ||||
|  | ||||
|           googleFontsStyleSheet = await ( | ||||
|             await fetch(googleFontHref(ctx.cfg.configuration.theme)) | ||||
|           ).text() | ||||
|  | ||||
|           while ((match = fontSourceRegex.exec(googleFontsStyleSheet)) !== null) { | ||||
|             // match[0] is the `url(path)`, match[1] is the `path` | ||||
|             const url = match[1] | ||||
|             // the static name of this file. | ||||
|             const [filename, ext] = url.split("/").pop()!.split(".") | ||||
|  | ||||
|             googleFontsStyleSheet = googleFontsStyleSheet.replace(url, `/fonts/${filename}.ttf`) | ||||
|  | ||||
|             promises.push( | ||||
|               fetch(url) | ||||
|                 .then((res) => { | ||||
|                   if (!res.ok) { | ||||
|                     throw new Error(`Failed to fetch font`) | ||||
|                   } | ||||
|                   return res.arrayBuffer() | ||||
|                 }) | ||||
|                 .then((buf) => | ||||
|                   write({ | ||||
|                     ctx, | ||||
|                     slug: joinSegments("fonts", filename) as FullSlug, | ||||
|                     ext: `.${ext}`, | ||||
|                     content: Buffer.from(buf), | ||||
|                   }), | ||||
|                 ), | ||||
|             ) | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       addGlobalPageResources(ctx, resources, componentResources) | ||||
|  | ||||
|       const stylesheet = joinStyles(ctx.cfg.configuration.theme, ...componentResources.css, styles) | ||||
|       const stylesheet = joinStyles( | ||||
|         ctx.cfg.configuration.theme, | ||||
|         ...componentResources.css, | ||||
|         googleFontsStyleSheet, | ||||
|         styles, | ||||
|       ) | ||||
|       const [prescript, postscript] = await Promise.all([ | ||||
|         joinScripts(componentResources.beforeDOMLoaded), | ||||
|         joinScripts(componentResources.afterDOMLoaded), | ||||
|       ]) | ||||
|  | ||||
|       const fps = await Promise.all([ | ||||
|       promises.push( | ||||
|         write({ | ||||
|           ctx, | ||||
|           slug: "index" as FullSlug, | ||||
| @@ -223,8 +268,9 @@ export const ComponentResources: QuartzEmitterPlugin<Options> = (opts?: Partial< | ||||
|           ext: ".js", | ||||
|           content: postscript, | ||||
|         }), | ||||
|       ]) | ||||
|       return fps | ||||
|       ) | ||||
|  | ||||
|       return await Promise.all(promises) | ||||
|     }, | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -7,7 +7,7 @@ type WriteOptions = { | ||||
|   ctx: BuildCtx | ||||
|   slug: FullSlug | ||||
|   ext: `.${string}` | "" | ||||
|   content: string | ||||
|   content: string | Buffer | ||||
| } | ||||
|  | ||||
| export const write = async ({ ctx, slug, ext, content }: WriteOptions): Promise<FilePath> => { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user