Added emoji support to Satori when generating OG images (#1593)
This commit is contained in:
parent
2acfa0fa23
commit
c97fd7089a
2 changed files with 82 additions and 1 deletions
|
@ -4,6 +4,7 @@ import { CSSResourceToStyleElement, JSResourceToScriptElement } from "../util/re
|
||||||
import { googleFontHref } from "../util/theme"
|
import { googleFontHref } from "../util/theme"
|
||||||
import { QuartzComponent, QuartzComponentConstructor, QuartzComponentProps } from "./types"
|
import { QuartzComponent, QuartzComponentConstructor, QuartzComponentProps } from "./types"
|
||||||
import satori, { SatoriOptions } from "satori"
|
import satori, { SatoriOptions } from "satori"
|
||||||
|
import { loadEmoji, getIconCode } from "../util/emoji"
|
||||||
import fs from "fs"
|
import fs from "fs"
|
||||||
import sharp from "sharp"
|
import sharp from "sharp"
|
||||||
import { ImageOptions, SocialImageOptions, getSatoriFont, defaultImage } from "../util/og"
|
import { ImageOptions, SocialImageOptions, getSatoriFont, defaultImage } from "../util/og"
|
||||||
|
@ -24,7 +25,21 @@ async function generateSocialImage(
|
||||||
// JSX that will be used to generate satori svg
|
// JSX that will be used to generate satori svg
|
||||||
const imageComponent = userOpts.imageStructure(cfg, userOpts, title, description, fonts, fileData)
|
const imageComponent = userOpts.imageStructure(cfg, userOpts, title, description, fonts, fileData)
|
||||||
|
|
||||||
const svg = await satori(imageComponent, { width, height, fonts })
|
const svg = await satori(imageComponent, {
|
||||||
|
width,
|
||||||
|
height,
|
||||||
|
fonts,
|
||||||
|
// `code` will be the detected language code, `emoji` if it's an Emoji, or `unknown` if not able to tell.
|
||||||
|
// `segment` will be the content to render.
|
||||||
|
loadAdditionalAsset: async (code: string, segment: string) => {
|
||||||
|
if (code === "emoji") {
|
||||||
|
// if segment is an emoji, load the image.
|
||||||
|
return `data:image/svg+xml;base64,${btoa(await loadEmoji("twemoji", getIconCode(segment)))}`
|
||||||
|
}
|
||||||
|
// if segment is normal text
|
||||||
|
return code
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
// Convert svg directly to webp (with additional compression)
|
// Convert svg directly to webp (with additional compression)
|
||||||
const compressed = await sharp(Buffer.from(svg)).webp({ quality: 40 }).toBuffer()
|
const compressed = await sharp(Buffer.from(svg)).webp({ quality: 40 }).toBuffer()
|
||||||
|
|
66
quartz/util/emoji.ts
Normal file
66
quartz/util/emoji.ts
Normal file
|
@ -0,0 +1,66 @@
|
||||||
|
/**
|
||||||
|
* Modified version of https://unpkg.com/twemoji@13.1.0/dist/twemoji.esm.js.
|
||||||
|
* Ported from https://github.com/vercel/satori/blob/48aea6f812365959c2888a25261c72ce17992c6d/playground/utils/twemoji.ts.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*! Copyright Twitter Inc. and other contributors. Licensed under MIT */
|
||||||
|
|
||||||
|
const U200D = String.fromCharCode(8205)
|
||||||
|
const UFE0Fg = /\uFE0F/g
|
||||||
|
|
||||||
|
export function getIconCode(char: string) {
|
||||||
|
return toCodePoint(char.indexOf(U200D) < 0 ? char.replace(UFE0Fg, "") : char)
|
||||||
|
}
|
||||||
|
|
||||||
|
function toCodePoint(unicodeSurrogates: string) {
|
||||||
|
const r = []
|
||||||
|
let c = 0,
|
||||||
|
p = 0,
|
||||||
|
i = 0
|
||||||
|
|
||||||
|
while (i < unicodeSurrogates.length) {
|
||||||
|
c = unicodeSurrogates.charCodeAt(i++)
|
||||||
|
if (p) {
|
||||||
|
r.push((65536 + ((p - 55296) << 10) + (c - 56320)).toString(16))
|
||||||
|
p = 0
|
||||||
|
} else if (55296 <= c && c <= 56319) {
|
||||||
|
p = c
|
||||||
|
} else {
|
||||||
|
r.push(c.toString(16))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return r.join("-")
|
||||||
|
}
|
||||||
|
|
||||||
|
export const apis = {
|
||||||
|
twemoji: (code: string) =>
|
||||||
|
"https://cdnjs.cloudflare.com/ajax/libs/twemoji/15.1.0/svg/" + code.toLowerCase() + ".svg",
|
||||||
|
openmoji: "https://cdn.jsdelivr.net/npm/@svgmoji/openmoji@3.2.0/svg/",
|
||||||
|
blobmoji: "https://cdn.jsdelivr.net/npm/@svgmoji/blob@3.2.0/svg/",
|
||||||
|
noto: "https://cdn.jsdelivr.net/gh/svgmoji/svgmoji/packages/svgmoji__noto/svg/",
|
||||||
|
fluent: (code: string) =>
|
||||||
|
"https://cdn.jsdelivr.net/gh/shuding/fluentui-emoji-unicode/assets/" +
|
||||||
|
code.toLowerCase() +
|
||||||
|
"_color.svg",
|
||||||
|
fluentFlat: (code: string) =>
|
||||||
|
"https://cdn.jsdelivr.net/gh/shuding/fluentui-emoji-unicode/assets/" +
|
||||||
|
code.toLowerCase() +
|
||||||
|
"_flat.svg",
|
||||||
|
}
|
||||||
|
|
||||||
|
const emojiCache: Record<string, Promise<any>> = {}
|
||||||
|
|
||||||
|
export function loadEmoji(type: keyof typeof apis, code: string) {
|
||||||
|
const key = type + ":" + code
|
||||||
|
if (key in emojiCache) return emojiCache[key]
|
||||||
|
|
||||||
|
if (!type || !apis[type]) {
|
||||||
|
type = "twemoji"
|
||||||
|
}
|
||||||
|
|
||||||
|
const api = apis[type]
|
||||||
|
if (typeof api === "function") {
|
||||||
|
return (emojiCache[key] = fetch(api(code)).then((r) => r.text()))
|
||||||
|
}
|
||||||
|
return (emojiCache[key] = fetch(`${api}${code.toUpperCase()}.svg`).then((r) => r.text()))
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue