Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/nberlette/migo

Generate dynamic OpenGraph images on Deno's Edge Network
https://github.com/nberlette/migo

api cloudflare-kv deno deno-deploy icns iconify migo og-image opengraph svg svg-to-png

Last synced: about 1 month ago
JSON representation

Generate dynamic OpenGraph images on Deno's Edge Network

Awesome Lists containing this project

README

        



migo


[![Click here for example OpenGraph Images][example-img-0]](#examples "Click
here for example OpenGraph Images")

---


### _Generate dynamic OpenGraph images on Deno's Edge Network_


[![Deploy with Deno][deploy-btn]][deploy]


## Schema

migo.deno.dev/:title.(png|svg)?:params

migo.deno.dev/:title/:subtitle.(png|svg)?:params

migo.deno.dev/:params/:title.(png|svg)

migo.deno.dev/:params/:title/:subtitle.(png|svg)


## Features

- [x] Just-in-time rendered, globally deployed via [**Deno Deploy**][deno]
- [x] Cached as immutable assets via [**Cloudflare**][kv] for fast loads.
- [x] Rendered as **`.svg`**, rasterized to **`.png`** via [**`resvg`**][resvg]
- [x] Integrated with 100,000 icons via [**Iconify**][iconify] + [**icns.ml**][icns]
- [x] [**Parameters**](#parameters) for fine-grained control of image colors,
dimensions, icon, and text.
- [ ] `TODO` create a friendly GUI for image creation ([see vercel][vercel])


## Formats

Every image is initially sculpted as an SVG ([Scalable Vector Graphics][svg]),
and you can optionally add the extension **`.svg`** to force that format in
the response.

Unfortunately most social media platforms don't support social images in SVG
format yet, so requests without an `.{svg,png}` extension are redirected to
`.png` prior to rendering.


## Icons

Icons are embedded from [**icns**][icns], another Deno-powered project of
mine. This means direct access to **over 100,000 icons**, and **millions of
color combinations**. A great tool to browse the available icons and make a
selection is [**icones**][icones] by [Anthony Fu][antfu].

To add an icon to an OG image, use the slug (in Iconify format) for the `icon`
param, like so:

> `icon={collection}:{icon}` (e.g. `?icon=tabler:brand-github`).

You can also use an override `iconUrl` parameter, with an encoded URI you'd
like to embed, e.g.:

> `icon=https%3A%2F%2Fcdn.jsdelivr.net%2Fgh%2Fremojansen%2Flogo.ts%40master%2Fts.svg`


## Parameters

There are numerous parameters you can use to control the look and feel of the
generated images. Parameters can be provided in either the first part of the
path or in the query string of the URL.


### Path (recommended)

For the best caching potential, I recommend only using the path-style
parameters on your images. Some CDN providers have unexpected caching behavior
when assets have query string parameters in their URI.

migo.deno.dev/bgColor=white&titleColor=black&icon=typescript/Title.png

> **Note**: Allowed delimiters are `&` (ampersand), `;` (semi-colon), or `::`.


### Query String

migo.deno.dev/title.png?bgColor=white&titleColor=black&icon=typescript

> **Note**: Query string params **must** use the `&` (ampersand) delimiter.


### Default Values

All available parameters and their default value (or formula used to calculate it):

```jsonc
// base props
width = 1280,
height = 640,
viewBox = "0 0 1280 640",
pxRatio = 1.5, // set to 1 for low-res
bgColor = "white",
borderRadius = 0, // rounded image corners

// icon
icon = "noto:t-rex", // set to false to disable icon
iconUrl = "https://icns.ml/{icon}.svg",
iconW = 240,
iconH = 240, // +iconW
iconX = 520, // ((width - iconW) / 2)
iconY = 60, // (iconH / 4)
iconColor = "black", // fill color
iconStroke = "none", // stroke color
iconStrokeWidth = 0, // stroke width

// title (first line of text)
titleX = 640, // (width / 2)
titleY = 450, // (iconH + iconY + (titleFontSize * 2.5))
titleFontSize = 64,
titleFontFamily = "sans-serif", // "Inter"
titleFontWeight = "bold",
titleColor = "#112233", // text color
titleStroke = "none", // stroke color
titleStrokeWidth = 0, // stroke width
titleTextAnchor = "middle", // text-anchor

// subtitle (second line of text)
subtitleX = 640, // (width / 2)
subtitleY = 530, // (titleY + (subtitleFontSize * 2.5))
subtitleFontSize = 32,
// "serif" | "sans-serif" | "monospace"
subtitleFontFamily = "monospace",
subtitleFontWeight = "normal",
subtitleColor = "#334455", // text color
subtitleStroke = "none", // stroke color
subtitleStrokeWidth = 0, // stroke width
subtitleTextAnchor = "middle" // text-anchor
```

---

## Performance

The only lag you might encounter is the very first time an image is requested
(this is unavoidable due to the render/raster steps). Thankfully, your users
should essentially never be the ones encountering that lag time; they get a
cache hit from their nearest edge datacenter.

---

## Examples

![Deno Module Starter][example-img-4]

![Nuxt Content Wind Starter][example-img-3]

![Creating Dynamic Social Cover Images][example-img-2]

---

[MIT] © [**Nicholas Berlette**][nberlette]

[MIT]: https://github.com/nberlette/migo/blob/main/LICENSE
[nberlette]: https://github.com/nberlette "Nicholas Berlette"
[icns]: https://icns.ml "icns - SVG as a Service"
[antfu]: https://github.com/antfu "Anthony Fu"
[icones]: https://icones.js.org "Browse every Iconify Collection with Icones"
[vercel]: https://og-image.vercel.app "Vercel's OG Image App"
[iconify]: https://iconify.design "Iconify Project Homepage"
[kv]: https://developers.cloudflare.com/workers/runtime-apis/kv "Cloudflare KV"
[svg]: https://w3.org/TR/SVG "SVG Specification from W3.org"
[resvg]: https://deno.land/x/resvg_wasm "Resvg Wasm"
[deno]: https://deno.com/deploy "Deno Deploy"
[deploy]: https://dash.deno.com/new?url=https%3a%2f%2fgithub.com%2fnberlette%2fmigo%2fraw%2fmain%2fmain.tsx "Deploy with Deno!"
[deploy-btn]:  "Deploy with Deno!"

[examples-splash]: https://migo.deno.dev/image.png?icon=deno&iconStrokeWidth=0.33&subtitleFontSize=48&iconColor=fff&bgColor=234&iconStroke=fff&titleColor=fff&subtitleColor=999&titleY=425&subtitleFontSize=36&title=Click%20here%20for%20example%20OG%20Images&subtitle=(or%20scroll%20down)&pxRatio=1&borderRadius=25
[example-img-0]: https://migo.deno.dev/image.png?title=migo.deno.dev&subtitle=Dynamic+OpenGraph+Images+on+Deno+Deploy&titleFontFamily=serif&titleFontSize=72&titleFontWeight=900&titleTextAnchor=left&titleX=160&titleY=110&subtitleFontSize=36&subtitleFontWeight=900&subtitleFontFamily=monospace&subtitleTextAnchor=left&subtitleX=40&subtitleY=250&pxRatio=1&width=1000&height=300&bgColor=111827&titleColor=fff&subtitleColor=ddd&icon=noto:t-rex&iconW=100&iconH=100&iconX=40&iconY=30&borderRadius=20
[example-img-1]: https://migo.deno.dev/image.png?icon=deno&iconStrokeWidth=0.33&subtitleFontSize=48&iconColor=fff&bgColor=234&iconStroke=fff&titleColor=fff&subtitleColor=papayawhip&titleY=425&subtitleFontSize=48&title=Edge-rendered%20OpenGraph%20Images&subtitle=migo.deno.dev&pxRatio=1&borderRadius=25&titleFontFamily=sans-serif
[example-img-2]: https://migo.deno.dev/image.png?icon=twitter&subtitleFontSize=48&iconColor=0cf&titleY=460&subtitleFontSize=48&title=Creating%20Dynamic%20Cover%20Images&subtitle=By%20Nicholas%20Berlette&borderRadius=25&titleFontFamily=monospace
[example-img-3]: https://migo.deno.dev/image.png?icon=nuxtdotjs&bgColor=112233&iconColor=00DC82&iconStroke=00DC82&iconStrokeWidth=0.55&titleColor=00DC82&subtitleColor=e0e0e0&iconW=300&iconY=50&titleY=460&title=Nuxt%20ContentWind%20Starter&subtitle=stackblitz.com%2fedit%2fcontent-wind&borderRadius=30&pxRatio=1
[example-img-4]: https://migo.deno.dev/image.png?title=deno911⁄+kit&subtitle=this+one+has+custom+alignment+and+size!&titleFontFamily=sans-serif&titleFontSize=80&titleFontWeight=900&titleTextAnchor=right&titleX=160&titleY=115&subtitleFontSize=36&subtitleFontWeight=900&subtitleFontFamily=monospace&subtitleTextAnchor=left&subtitleX=40&subtitleY=260&pxRatio=1&width=1000&height=300&bgColor=fff&titleColor=123&subtitleColor=456&icon=noto:sauropod&iconW=100&iconH=100&iconX=40&iconY=30&borderRadius=20