Perf: reuse Shiki highlighters per theme/lang (#3130)
* reuse Shiki highlighters per theme/lang * chore: adding changeset
This commit is contained in:
parent
9e35758ec3
commit
394ab90547
3 changed files with 28 additions and 1 deletions
5
.changeset/eleven-guests-rhyme.md
Normal file
5
.changeset/eleven-guests-rhyme.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
'astro': patch
|
||||||
|
---
|
||||||
|
|
||||||
|
Updates `<Code />` component to cache and reuse Shiki highlighters
|
|
@ -1,6 +1,6 @@
|
||||||
---
|
---
|
||||||
import type * as shiki from 'shiki';
|
import type * as shiki from 'shiki';
|
||||||
import { getHighlighter } from 'shiki';
|
import { getHighlighter } from './Shiki.js';
|
||||||
|
|
||||||
export interface Props {
|
export interface Props {
|
||||||
/** The code to highlight. Required. */
|
/** The code to highlight. Required. */
|
||||||
|
|
22
packages/astro/components/Shiki.js
Normal file
22
packages/astro/components/Shiki.js
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
import { getHighlighter as getShikiHighlighter } from 'shiki';
|
||||||
|
|
||||||
|
// Caches Promise<Highligher> for reuse when the same theme and langs are provided
|
||||||
|
const _resolvedHighlighters = new Map();
|
||||||
|
|
||||||
|
function stringify(opts) {
|
||||||
|
// Always sort keys before stringifying to make sure objects match regardless of parameter ordering
|
||||||
|
return JSON.stringify(opts, Object.keys(opts).sort());
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getHighlighter(opts) {
|
||||||
|
const key = stringify(opts);
|
||||||
|
|
||||||
|
// Highlighter has already been requested, reuse the same instance
|
||||||
|
if (_resolvedHighlighters.has(key)) { return _resolvedHighlighters.get(key) }
|
||||||
|
|
||||||
|
// Start the async getHighlighter call and cache the Promise
|
||||||
|
const highlighter = getShikiHighlighter(opts);
|
||||||
|
_resolvedHighlighters.set(key, highlighter);
|
||||||
|
|
||||||
|
return highlighter;
|
||||||
|
}
|
Loading…
Reference in a new issue