sass-site/source/helpers/function.ts

94 lines
3.1 KiB
TypeScript
Raw Normal View History

2023-05-27 01:11:28 +02:00
import * as cheerio from 'cheerio';
import stripIndent from 'strip-indent';
2023-05-27 01:11:28 +02:00
2023-06-19 23:55:26 +02:00
import {codeBlock} from './components';
import {liquidEngine} from './engines';
2023-05-27 01:11:28 +02:00
const links: Record<string, string> = {
number: '/documentation/values/numbers',
string: '/documentation/values/strings',
'quoted string': '/documentation/values/strings#quoted',
'unquoted string': '/documentation/values/strings#unquoted',
color: '/documentation/values/colors',
list: '/documentation/values/lists',
map: '/documentation/values/maps',
boolean: '/documentation/values/booleans',
null: '/documentation/values/null',
function: '/documentation/values/functions',
selector: '/documentation/modules/selector#selector-values',
};
const returnTypeLink = (returnType: string) =>
returnType
.split('|')
2023-06-19 23:55:26 +02:00
.map(type => {
2023-05-27 01:11:28 +02:00
type = type.trim();
const link = links[type];
if (!link) {
throw new Error(`Unknown type ${type}`);
}
return `<a href="${link}">${type}</a>`;
})
.join(' | ');
/** Renders API docs for a Sass function (or mixin).
*
* The function's name is parsed from the signature. The API description is
* passed as an HTML block. If `returns:type` is passed as the last argument,
2023-05-27 01:11:28 +02:00
* it's included as the function's return type.
*
* Multiple signatures may be passed, in which case they're all included in
* sequence.
*/
export function _function(content: string, ...signatures: string[]) {
// Parse the last argument as the return type, if it's present
const returns = signatures.at(-1)?.match(/returns?:\s*(.*)/)?.[1];
if (returns) {
signatures.pop();
}
2023-05-27 01:11:28 +02:00
// Highlight each signature
const names: string[] = [];
2023-06-19 23:55:26 +02:00
const highlightedSignatures = signatures.map(signature => {
signature = stripIndent(signature).trim();
2023-05-27 01:11:28 +02:00
const [name] = signature.split('(', 2);
const nameWithoutNamespace = name.split('.').at(-1) || name;
const html = codeBlock(`@function ${signature}`, 'scss');
const $ = cheerio.load(html);
const signatureElements = $('pre code')
.contents()
.filter((index, element) => $(element).text() !== '@function');
// Add a class to make it easier to index function documentation.
2023-05-27 01:11:28 +02:00
if (!names.includes(nameWithoutNamespace)) {
names.push(nameWithoutNamespace);
const nameEl = signatureElements
.filter((index, element) => {
2023-06-19 23:55:26 +02:00
return $(element).text() === nameWithoutNamespace;
2023-05-27 01:11:28 +02:00
})
.eq(0);
nameEl.addClass('docSearch-function');
nameEl.attr('name', name);
}
return signatureElements
.toArray()
2023-06-19 23:55:26 +02:00
.map(el => $.html(el))
2023-05-27 01:11:28 +02:00
.join('')
2023-05-27 02:23:55 +02:00
.trim();
2023-05-27 01:11:28 +02:00
});
// Render the final HTML
return liquidEngine.renderFile('function', {
names,
signatures: highlightedSignatures.join('\n'),
content,
returns: returns ? returnTypeLink(returns) : null,
2023-05-27 01:11:28 +02:00
});
}
/* eslint-disable @typescript-eslint/no-unsafe-member-access,
@typescript-eslint/no-unsafe-call,
@typescript-eslint/no-explicit-any */
export default function typePlugin(eleventyConfig: any) {
eleventyConfig.addPairedLiquidShortcode('function', _function);
}