Skip to content

Commit 6d90ef7

Browse files
committedDec 4, 2023
feat: use unocss for inlining tw classes with config extending
1 parent 7cc25fe commit 6d90ef7

File tree

20 files changed

+567
-44
lines changed

20 files changed

+567
-44
lines changed
 

‎README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ Generate OG Images with Vue templates in Nuxt.
3434

3535
- ✨ Create an `og:image` using the built-in templates or make your own with Vue components
3636
- 🎨 Design and test your `og:image` in the Nuxt DevTools OG Image Playground with full HMR
37-
- ▲ Render using [Satori](https://github.com/vercel/satori): Tailwind classes, Google fonts, emoji support and more!
37+
- ▲ Render using [Satori](https://github.com/vercel/satori): Tailwind / UnoCSS with your theme, Google fonts, 6 emoji families supported and more!
3838
- 🤖 Or prerender using the Browser: Supporting painless, complex templates
3939
- 📸 Feeling lazy? Just generate screenshots for every page: hide elements, wait for animations, and more
4040
- ⚙️ Works on the edge: Vercel Edge, Netlify Edge and Cloudflare Workers

‎package.json

+4
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,8 @@
4545
"@nuxt/kit": "^3.8.2",
4646
"@resvg/resvg-js": "^2.6.0",
4747
"@resvg/resvg-wasm": "^2.6.0",
48+
"@unocss/core": "^0.58.0",
49+
"@unocss/preset-wind": "^0.58.0",
4850
"chrome-launcher": "^1.1.0",
4951
"css-inline": "^0.11.0",
5052
"defu": "^6.1.3",
@@ -78,6 +80,8 @@
7880
"@nuxt/ui": "^2.11.0",
7981
"@nuxtjs/eslint-config-typescript": "^12.1.0",
8082
"@nuxtjs/i18n": "8.0.0-rc.6",
83+
"@nuxtjs/tailwindcss": "^6.10.1",
84+
"@unocss/nuxt": "^0.58.0",
8185
"bumpp": "^9.2.0",
8286
"eslint": "8.55.0",
8387
"jest-image-snapshot": "^6.3.0",

‎pnpm-lock.yaml

+305
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎src/module.ts

+17
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ import { hash } from 'ohash'
1919
import { relative } from 'pathe'
2020
import type { ResvgRenderOptions } from '@resvg/resvg-js'
2121
import type { SharpOptions } from 'sharp'
22+
import { defu } from 'defu'
2223
import { version } from '../package.json'
2324
import type { FontConfig, InputFontConfig, OgImageComponent, OgImageOptions, OgImageRuntimeConfig } from './runtime/types'
2425
import { type RuntimeCompatibilitySchema, getPresetNitroPresetCompatibility, resolveNitroPreset } from './compatibility'
@@ -221,6 +222,7 @@ export default defineNuxtModule<ModuleOptions>({
221222
await addComponentsDir({
222223
path: resolve('./runtime/components/Templates/Community'),
223224
island: true,
225+
watch: true,
224226
})
225227

226228
;[
@@ -294,6 +296,21 @@ export default defineNuxtModule<ModuleOptions>({
294296
return `export const componentNames = ${JSON.stringify(ogImageComponentCtx.components)}`
295297
}
296298

299+
// support simple theme extends
300+
let unoCssConfig: any = { theme: {} }
301+
// @ts-ignore runtime type
302+
nuxt.hook('tailwindcss:config', (tailwindConfig) => {
303+
// @ts-expect-error untyped
304+
unoCssConfig = defu(tailwindConfig.theme.extend, { ...(tailwindConfig.theme || {}), extend: undefined })
305+
})
306+
// @ts-expect-error runtime type
307+
nuxt.hook('unocss:config', (_unoCssConfig) => {
308+
unoCssConfig = { ..._unoCssConfig.theme }
309+
})
310+
nuxt.options.nitro.virtual['#nuxt-og-image/unocss-config.mjs'] = () => {
311+
return `export const theme = ${JSON.stringify(unoCssConfig)}`
312+
}
313+
297314
extendTypes('nuxt-og-image', ({ typesPath }) => {
298315
// need to map our components to types so we can import them
299316
const componentImports = ogImageComponentCtx.components.map((component) => {

‎src/runtime/components/Templates/Community/UnJs.vue

+13-5
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,9 @@ const description = computed(() => (props.description || '').slice(0, 200))
4444
<span v-if="repo" class="ml-2">/</span>
4545
<span v-if="repo" class="ml-2 font-bold">{{ repo }}</span>
4646
</h1>
47-
<p class="text-gray-500 max-w-[700px] text-[35px] leading-[60px]">{{ description }}</p>
47+
<p class="text-gray-500 max-w-[700px] text-[35px] leading-[60px]">
48+
{{ description }}
49+
</p>
4850
</div>
4951
<div class="text-[200px]">
5052
{{ emoji }}
@@ -56,21 +58,27 @@ const description = computed(() => (props.description || '').slice(0, 200))
5658
<svg xmlns="http://www.w3.org/2000/svg" width="1em" height="1em" viewBox="0 0 32 32"><path fill="#888888" d="m4.67 28l6.39-12l7.3 6.49a2 2 0 0 0 1.7.47a2 2 0 0 0 1.42-1.07L27 10.9l-1.82-.9l-5.49 11l-7.3-6.49a2 2 0 0 0-1.68-.51a2 2 0 0 0-1.42 1L4 25V2H2v26a2 2 0 0 0 2 2h26v-2Z" /></svg>
5759
<div class="pl-2">
5860
<div>{{ downloads }}</div>
59-
<div class="text-lg text-gray-600">Monthly Downloads</div>
61+
<div class="text-lg text-gray-600">
62+
Monthly Downloads
63+
</div>
6064
</div>
6165
</div>
6266
<div class="flex flex-row pr-10">
6367
<svg xmlns="http://www.w3.org/2000/svg" width="1em" height="1em" viewBox="0 0 32 32"><path fill="#888888" d="m16 6.52l2.76 5.58l.46 1l1 .15l6.16.89l-4.38 4.3l-.75.73l.18 1l1.05 6.13l-5.51-2.89L16 23l-.93.49l-5.51 2.85l1-6.13l.18-1l-.74-.77l-4.42-4.35l6.16-.89l1-.15l.46-1L16 6.52M16 2l-4.55 9.22l-10.17 1.47l7.36 7.18L6.9 30l9.1-4.78L25.1 30l-1.74-10.13l7.36-7.17l-10.17-1.48Z" /></svg>
6468
<div class="pl-2">
6569
<div>{{ stars }}</div>
66-
<div class="text-lg text-gray-600">Stars</div>
70+
<div class="text-lg text-gray-600">
71+
Stars
72+
</div>
6773
</div>
6874
</div>
6975
<div class="flex flex-row pr-10">
70-
<svg xmlns="http://www.w3.org/2000/svg" width="1em" height="1em" viewBox="0 0 32 32"><path fill="#888888" d="M22.45 6a5.47 5.47 0 0 1 3.91 1.64a5.7 5.7 0 0 1 0 8L16 26.13L5.64 15.64a5.7 5.7 0 0 1 0-8a5.48 5.48 0 0 1 7.82 0l2.54 2.6l2.53-2.58A5.44 5.44 0 0 1 22.45 6m0-2a7.47 7.47 0 0 0-5.34 2.24L16 7.36l-1.11-1.12a7.49 7.49 0 0 0-10.68 0a7.72 7.72 0 0 0 0 10.82L16 29l11.79-11.94a7.72 7.72 0 0 0 0-10.82A7.49 7.49 0 0 0 22.45 4Z"/></svg>
76+
<svg xmlns="http://www.w3.org/2000/svg" width="1em" height="1em" viewBox="0 0 32 32"><path fill="#888888" d="M22.45 6a5.47 5.47 0 0 1 3.91 1.64a5.7 5.7 0 0 1 0 8L16 26.13L5.64 15.64a5.7 5.7 0 0 1 0-8a5.48 5.48 0 0 1 7.82 0l2.54 2.6l2.53-2.58A5.44 5.44 0 0 1 22.45 6m0-2a7.47 7.47 0 0 0-5.34 2.24L16 7.36l-1.11-1.12a7.49 7.49 0 0 0-10.68 0a7.72 7.72 0 0 0 0 10.82L16 29l11.79-11.94a7.72 7.72 0 0 0 0-10.82A7.49 7.49 0 0 0 22.45 4Z" /></svg>
7177
<div class="pl-2">
7278
<div>{{ contributors }}</div>
73-
<div class="text-lg text-gray-600">Contributors</div>
79+
<div class="text-lg text-gray-600">
80+
Contributors
81+
</div>
7482
</div>
7583
</div>
7684
</div>

‎src/runtime/components/Templates/Community/Wave.vue

-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@
33
* @credits Full Stack Heroes <https://fullstackheroes.com/>
44
*/
55
6-
76
withDefaults(defineProps<{
87
title?: string
98
}>(), {

‎src/runtime/core/html/devIframeTemplate.ts

+14-8
Original file line numberDiff line numberDiff line change
@@ -4,10 +4,11 @@ import type { FontConfig, H3EventOgImageRender } from '../../types'
44
import { useOgImageRuntimeConfig } from '../../utils'
55
import { applyEmojis } from './applyEmojis'
66
import { fetchIsland } from './fetchIsland'
7+
import { theme } from '#nuxt-og-image/unocss-config.mjs'
78

89
export async function devIframeTemplate(ctx: H3EventOgImageRender) {
910
const { options } = ctx
10-
const { fonts, satoriOptions } = useOgImageRuntimeConfig()
11+
const { fonts } = useOgImageRuntimeConfig()
1112
// const path = options.path
1213
// const scale = query.scale
1314
// const mode = query.mode || 'light'
@@ -81,15 +82,20 @@ svg[data-emoji] {
8182
],
8283
script: [
8384
{
84-
src: 'https://cdn.tailwindcss.com',
85+
src: 'https://cdn.jsdelivr.net/npm/@unocss/runtime/preset-wind.global.js',
8586
},
8687
{
87-
innerHTML: `tailwind.config = {
88-
corePlugins: {
89-
preflight: false,
90-
},
91-
theme: ${JSON.stringify(satoriOptions?.tailwindConfig?.theme || {})}
92-
}`,
88+
innerHTML: `
89+
window.__unocss = {
90+
theme: ${JSON.stringify(theme)},
91+
presets: [
92+
() => window.__unocss_runtime.presets.presetWind(),
93+
],
94+
}
95+
`,
96+
},
97+
{
98+
src: 'https://cdn.jsdelivr.net/npm/@unocss/runtime/core.global.js',
9399
},
94100
],
95101
link: [
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
import { createGenerator } from '@unocss/core'
2+
import presetWind from '@unocss/preset-wind'
3+
import type { VNode } from '../../../../types'
4+
import { defineSatoriTransformer } from '../utils'
5+
import { theme } from '#nuxt-og-image/unocss-config.mjs'
6+
7+
const uno = createGenerator({ theme }, {
8+
presets: [
9+
presetWind(),
10+
],
11+
})
12+
13+
// convert classes to inline style using unocss, provides more robust API than satori
14+
export default defineSatoriTransformer({
15+
filter: (node: VNode) => !!node.props?.class,
16+
transform: async (node: VNode) => {
17+
const classes: string = node.props.class || ''
18+
// normalise the styles
19+
const styles = node.props.style as Record<string, string> || {}
20+
21+
const replacedClasses = new Set()
22+
for (const token of classes.split(' ').filter(c => c.trim())) {
23+
const parsedToken = await uno.parseToken(token)
24+
if (parsedToken) {
25+
const inlineStyles = parsedToken[0][2].split(';').filter(s => !!s?.trim())
26+
const vars: Record<string, string> = {}
27+
inlineStyles.filter(s => s.startsWith('--'))
28+
.forEach((s) => {
29+
const [key, value] = s.split(':')
30+
vars[key] = value
31+
})
32+
inlineStyles.filter(s => !s.startsWith('--'))
33+
.forEach((s) => {
34+
const [key, value] = s.split(':')
35+
const camelCasedKey = key.replace(/-([a-z])/g, (_, letter) => letter.toUpperCase())
36+
// we need to replace any occurances of a var key with the var values, avoid replacing existing inline styles
37+
if (!styles[camelCasedKey])
38+
styles[camelCasedKey] = value.replace(/var\((.*?)\)/g, (_, k) => vars[k.trim()])
39+
})
40+
replacedClasses.add(token)
41+
}
42+
}
43+
node.props.class = classes.split(' ').filter(c => !replacedClasses.has(c)).join(' ')
44+
node.props.tw = classes.split(' ').filter(c => !replacedClasses.has(c)).join(' ')
45+
node.props.style = styles
46+
},
47+
})

‎src/runtime/core/renderers/satori/vnodes.ts

+2
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import { fetchIsland } from '../../html/fetchIsland'
44
import { applyInlineCss } from '../../html/applyInlineCss'
55
import { applyEmojis } from '../../html/applyEmojis'
66
import { walkSatoriTree } from './utils'
7+
import unocss from './plugins/unocss'
78
import emojis from './plugins/emojis'
89
import twClasses from './plugins/twClasses'
910
import imageSrc from './plugins/imageSrc'
@@ -26,6 +27,7 @@ export async function createVNodes(ctx: H3EventOgImageRender): Promise<VNode> {
2627
const satoriTree = convertHtmlToSatori(template)
2728
// process the tree
2829
await walkSatoriTree(ctx, satoriTree, [
30+
unocss,
2931
emojis,
3032
twClasses,
3133
imageSrc,

‎test/fixtures/tailwind/app.vue

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
<script lang="ts" setup>
2+
import { defineOgImage } from '#imports'
3+
4+
defineOgImage({
5+
component: 'CustomClasses',
6+
})
7+
</script>
8+
9+
<template>
10+
<div>hello world</div>
11+
</template>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
<template>
2+
<div class="w-full h-full flex items-center justify-center bg-primary-500 text-base">
3+
<h1 class="text-mega-big">
4+
Hello World
5+
</h1>
6+
</div>
7+
</template>

‎test/fixtures/tailwind/nuxt.config.ts

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
import NuxtOgImage from '../../../src/module'
2+
3+
// https://v3.nuxtjs.org/api/configuration/nuxt.config
4+
export default defineNuxtConfig({
5+
modules: [
6+
NuxtOgImage,
7+
'@nuxtjs/tailwindcss',
8+
],
9+
tailwindcss: {
10+
exposeConfig: true,
11+
},
12+
site: {
13+
url: 'https://nuxtseo.com',
14+
},
15+
devtools: { enabled: false },
16+
debug: process.env.NODE_ENV === 'test',
17+
})
+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
import type { Config } from 'tailwindcss'
2+
import colors from 'tailwindcss/colors'
3+
4+
export default <Partial<Config>>{
5+
theme: {
6+
extend: {
7+
fontSize: {
8+
'mega-big': '100px',
9+
},
10+
colors: {
11+
base: colors.white,
12+
primary: colors.green,
13+
},
14+
},
15+
},
16+
}

‎test/fixtures/unocss/app.vue

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
<script lang="ts" setup>
2+
import { defineOgImage } from '#imports'
3+
4+
defineOgImage({
5+
component: 'CustomClasses',
6+
})
7+
</script>
8+
9+
<template>
10+
<div>hello world</div>
11+
</template>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
<template>
2+
<div class="w-full h-full flex items-center justify-center bg-primary-500 text-base">
3+
<h1 class="text-mega-big">
4+
Hello World
5+
</h1>
6+
</div>
7+
</template>

‎test/fixtures/unocss/nuxt.config.ts

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
import NuxtOgImage from '../../../src/module'
2+
3+
// https://v3.nuxtjs.org/api/configuration/nuxt.config
4+
export default defineNuxtConfig({
5+
modules: [
6+
NuxtOgImage,
7+
'@unocss/nuxt',
8+
],
9+
site: {
10+
url: 'https://nuxtseo.com',
11+
},
12+
devtools: { enabled: false },
13+
debug: process.env.NODE_ENV === 'test',
14+
})

‎test/fixtures/unocss/uno.config.ts

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
import colors from 'tailwindcss/colors'
2+
import { defineConfig } from 'unocss'
3+
4+
export default defineConfig({
5+
theme: {
6+
fontSize: {
7+
'mega-big': '100px',
8+
},
9+
colors: {
10+
base: colors.white,
11+
primary: colors.green,
12+
},
13+
},
14+
})

‎test/integration/endpoints/satori/html.test.ts

+11-8
Original file line numberDiff line numberDiff line change
@@ -33,16 +33,19 @@ describe('html', () => {
3333
display: inline-block;
3434
}
3535
</style>
36-
<script src=\\"https://cdn.tailwindcss.com\\"></script>
37-
<script>tailwind.config = {
38-
corePlugins: {
39-
preflight: false,
40-
},
41-
theme: {}
42-
}</script>
36+
<script src=\\"https://cdn.jsdelivr.net/npm/@unocss/runtime/preset-wind.global.js\\"></script>
37+
<script>
38+
window.__unocss = {
39+
theme: {\\"theme\\":{}},
40+
presets: [
41+
() => window.__unocss_runtime.presets.presetWind(),
42+
],
43+
}
44+
</script>
45+
<script src=\\"https://cdn.jsdelivr.net/npm/@unocss/runtime/core.global.js\\"></script>
4346
<link href=\\"https://cdn.jsdelivr.net/npm/gardevoir\\" rel=\\"stylesheet\\">
4447
<link href=\\"https://fonts.googleapis.com/css2?family=Inter:wght@400;700&display=swap\\" rel=\\"stylesheet\\"></head>
45-
<body ><div data-v-inspector-ignore=\\"true\\" style=\\"position: relative; display: flex; margin: 0 auto; width: 1200px; height: 600px; overflow: hidden;\\"><div class=\\"w-full h-full flex justify-between relative p-[60px] bg-white text-gray-900\\"><div class=\\"flex absolute top-0 right-[-100%]\\" style=\\"width:200%;height:200%;background-image:radial-gradient(circle, rgba(0, 220, 130, 0.5) 0%, rgba(255, 255, 255, 0.7) 50%, rgba(255, 255, 255, 0) 70%);\\"></div><div class=\\"h-full w-full justify-between relative\\"><div class=\\"flex flex-row justify-between items-center\\"><div class=\\"flex flex-col w-full\\" style=\\"\\"><h1 class=\\"font-bold mb-[30px] text-[75px] max-w-[70%]\\">Hello World</h1><!----></div><!----></div><div class=\\"flex flex-row justify-center items-center text-left w-full\\"><!--[--><svg height=\\"50\\" width=\\"50\\" class=\\"mr-3\\" viewBox=\\"0 0 200 200\\" xmlns=\\"http://www.w3.org/2000/svg\\"><path fill=\\"#00dc82\\" d=\\"M62.3,-53.9C74.4,-34.5,73.5,-9,67.1,13.8C60.6,36.5,48.7,56.5,30.7,66.1C12.7,75.7,-11.4,74.8,-31.6,65.2C-51.8,55.7,-67.9,37.4,-73.8,15.7C-79.6,-6,-75.1,-31.2,-61.1,-51C-47.1,-70.9,-23.6,-85.4,0.8,-86C25.1,-86.7,50.2,-73.4,62.3,-53.9Z\\" transform=\\"translate(100 100)\\"></path></svg><p style=\\"font-size:25px;\\" class=\\"font-bold\\">nuxt-og-image</p><!--]--></div></div></div></div></body>
48+
<body ><div data-v-inspector-ignore=\\"true\\" style=\\"position: relative; display: flex; margin: 0 auto; width: 1200px; height: 600px; overflow: hidden;\\"><div class=\\"w-full h-full flex justify-between relative p-[60px] bg-white text-gray-900\\" nuxt-ssr-component-uid><div class=\\"flex absolute top-0 right-[-100%]\\" style=\\"width:200%;height:200%;background-image:radial-gradient(circle, rgba(0, 220, 130, 0.5) 0%, rgba(255, 255, 255, 0.7) 50%, rgba(255, 255, 255, 0) 70%);\\"></div><div class=\\"h-full w-full justify-between relative\\"><div class=\\"flex flex-row justify-between items-center\\"><div class=\\"flex flex-col w-full\\" style=\\"\\"><h1 class=\\"font-bold mb-[30px] text-[75px] max-w-[70%]\\">Hello World</h1><!----></div><!----></div><div class=\\"flex flex-row justify-center items-center text-left w-full\\"><!--[--><svg height=\\"50\\" width=\\"50\\" class=\\"mr-3\\" viewBox=\\"0 0 200 200\\" xmlns=\\"http://www.w3.org/2000/svg\\"><path fill=\\"#00dc82\\" d=\\"M62.3,-53.9C74.4,-34.5,73.5,-9,67.1,13.8C60.6,36.5,48.7,56.5,30.7,66.1C12.7,75.7,-11.4,74.8,-31.6,65.2C-51.8,55.7,-67.9,37.4,-73.8,15.7C-79.6,-6,-75.1,-31.2,-61.1,-51C-47.1,-70.9,-23.6,-85.4,0.8,-86C25.1,-86.7,50.2,-73.4,62.3,-53.9Z\\" transform=\\"translate(100 100)\\"></path></svg><p style=\\"font-size:25px;\\" class=\\"font-bold\\">nuxt-og-image</p><!--]--></div></div></div></div></body>
4649
</html>"
4750
`)
4851
}, 60000)

‎test/integration/endpoints/satori/json.test.ts

+55-20
Large diffs are not rendered by default.

‎test/integration/endpoints/satori/svg.test.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ describe('svg', () => {
1313
expect(svg).toMatchInlineSnapshot(`
1414
Blob {
1515
Symbol(kHandle): Blob {},
16-
Symbol(kLength): 14732,
16+
Symbol(kLength): 14766,
1717
Symbol(kType): "image/svg+xml",
1818
}
1919
`)

0 commit comments

Comments
 (0)
Please sign in to comment.