astro/packages/astro-prism/src/plugin.ts

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

188 lines
5.1 KiB
TypeScript
Raw Normal View History

export function addAstro(Prism: typeof import('prismjs')) {
2021-07-20 04:07:49 +00:00
if (Prism.languages.astro) {
return;
}
2021-12-22 21:11:05 +00:00
let scriptLang: string;
2021-07-20 04:07:49 +00:00
if (Prism.languages.typescript) {
scriptLang = 'typescript';
} else {
scriptLang = 'javascript';
// eslint-disable-next-line no-console
console.warn(
'Prism TypeScript language not loaded, Astro scripts will be treated as JavaScript.'
);
}
2021-12-22 21:11:05 +00:00
let script = Prism.util.clone(Prism.languages[scriptLang]);
2021-12-22 21:11:05 +00:00
let space = /(?:\s|\/\/.*(?!.)|\/\*(?:[^*]|\*(?!\/))\*\/)/.source;
let braces = /(?:\{(?:\{(?:\{[^{}]*\}|[^{}])*\}|[^{}])*\})/.source;
let spread = /(?:\{<S>*\.{3}(?:[^{}]|<BRACES>)*\})/.source;
2021-12-22 21:11:05 +00:00
function re(source: string, flags?: string) {
source = source
2021-07-20 04:07:49 +00:00
.replace(/<S>/g, function () {
return space;
})
.replace(/<BRACES>/g, function () {
return braces;
})
.replace(/<SPREAD>/g, function () {
return spread;
});
return RegExp(source, flags);
}
2021-12-22 21:11:05 +00:00
spread = re(spread).source;
2021-12-22 21:11:05 +00:00
Prism.languages.astro = Prism.languages.extend('markup', script);
(Prism.languages.astro as any).tag.pattern = re(
2021-07-20 04:07:49 +00:00
/<\/?(?:[\w.:-]+(?:<S>+(?:[\w.:$-]+(?:=(?:"(?:\\[^]|[^\\"])*"|'(?:\\[^]|[^\\'])*'|[^\s{'"/>=]+|<BRACES>))?|<SPREAD>))*<S>*\/?)?>/
.source
);
2021-12-22 21:11:05 +00:00
(Prism.languages.astro as any).tag.inside['tag'].pattern = /^<\/?[^\s>\/]*/i;
(Prism.languages.astro as any).tag.inside['attr-value'].pattern =
/=(?!\{)(?:"(?:\\[^]|[^\\"])*"|'(?:\\[^]|[^\\'])*'|[^\s'">]+)/i;
(Prism.languages.astro as any).tag.inside['tag'].inside['class-name'] =
/^[A-Z]\w*(?:\.[A-Z]\w*)*$/;
(Prism.languages.astro as any).tag.inside['comment'] = script['comment'];
2021-12-22 21:11:05 +00:00
2021-07-20 04:07:49 +00:00
Prism.languages.insertBefore(
'inside',
'attr-name',
{
spread: {
pattern: re(/<SPREAD>/.source),
inside: Prism.languages.astro,
},
},
(Prism.languages.astro as any).tag
2021-07-20 04:07:49 +00:00
);
2021-12-22 21:11:05 +00:00
2021-07-20 04:07:49 +00:00
Prism.languages.insertBefore(
'inside',
'special-attr',
{
script: {
// Allow for two levels of nesting
pattern: re(/=<BRACES>/.source),
inside: {
'script-punctuation': {
pattern: /^=(?={)/,
alias: 'punctuation',
},
rest: Prism.languages.astro,
},
2021-07-20 04:07:49 +00:00
alias: `language-${scriptLang}`,
},
2021-07-20 04:07:49 +00:00
},
(Prism.languages.astro as any).tag
2021-07-20 04:07:49 +00:00
);
2021-12-22 21:11:05 +00:00
// The following will handle plain text inside tags
let stringifyToken = function (token: any) {
if (!token) {
return '';
}
if (typeof token === 'string') {
return token;
}
if (typeof token.content === 'string') {
return token.content;
}
return token.content.map(stringifyToken).join('');
};
2021-12-22 21:11:05 +00:00
let walkTokens = function (tokens: any) {
let openedTags: any[] = [];
for (let i = 0; i < tokens.length; i++) {
let token = tokens[i];
2021-12-22 21:11:05 +00:00
// This breaks styles, not sure why
2021-07-20 04:07:49 +00:00
if (token.type === 'style') {
return;
}
2021-12-22 21:11:05 +00:00
let notTagNorBrace = false;
2021-12-22 21:11:05 +00:00
if (typeof token !== 'string') {
if (token.type === 'tag' && token.content[0] && token.content[0].type === 'tag') {
// We found a tag, now find its kind
2021-12-22 21:11:05 +00:00
if (token.content[0].content[0].content === '</') {
// Closing tag
if (
openedTags.length > 0 &&
openedTags[openedTags.length - 1].tagName ===
stringifyToken(token.content[0].content[1])
) {
// Pop matching opening tag
openedTags.pop();
}
} else {
if (token.content[token.content.length - 1].content === '/>') {
// Autoclosed tag, ignore
} else {
// Opening tag
openedTags.push({
tagName: stringifyToken(token.content[0].content[1]),
2021-07-20 04:07:49 +00:00
openedBraces: 0,
});
}
}
} else if (openedTags.length > 0 && token.type === 'punctuation' && token.content === '{') {
// Here we might have entered a Astro context inside a tag
openedTags[openedTags.length - 1].openedBraces++;
} else if (
openedTags.length > 0 &&
openedTags[openedTags.length - 1].openedBraces > 0 &&
token.type === 'punctuation' &&
token.content === '}'
) {
// Here we might have left a Astro context inside a tag
openedTags[openedTags.length - 1].openedBraces--;
} else {
2021-07-20 04:07:49 +00:00
notTagNorBrace = true;
}
}
if (notTagNorBrace || typeof token === 'string') {
if (openedTags.length > 0 && openedTags[openedTags.length - 1].openedBraces === 0) {
// Here we are inside a tag, and not inside a Astro context.
// That's plain text: drop any tokens matched.
let plainText = stringifyToken(token);
2021-12-22 21:11:05 +00:00
// And merge text with adjacent text
if (
i < tokens.length - 1 &&
(typeof tokens[i + 1] === 'string' || tokens[i + 1].type === 'plain-text')
) {
plainText += stringifyToken(tokens[i + 1]);
tokens.splice(i + 1, 1);
}
if (i > 0 && (typeof tokens[i - 1] === 'string' || tokens[i - 1].type === 'plain-text')) {
plainText = stringifyToken(tokens[i - 1]) + plainText;
tokens.splice(i - 1, 1);
i--;
}
2021-12-22 21:11:05 +00:00
tokens[i] = new Prism.Token('plain-text', plainText, undefined, plainText);
}
}
2021-12-22 21:11:05 +00:00
if (token.content && typeof token.content !== 'string') {
walkTokens(token.content);
}
}
};
2021-12-22 21:11:05 +00:00
Prism.hooks.add('after-tokenize', function (env: any) {
if (env.language !== 'astro') {
return;
}
walkTokens(env.tokens);
});
2021-07-20 04:07:49 +00:00
}