--- import type { HTMLAttributes } from 'astro/types'; import { parse } from 'node-html-parser'; type _ModIcons = | 'oneconfig' | 'chatting' | 'polysprint' | 'vanillahud' | 'overflowanimations' | 'crashpatch' | 'polytime' | 'polyweather' | 'keystrokes'; export type ModIcons = _ModIcons; // bypass for Astro compiler issue https://github.com/withastro/compiler/issues/554#issuecomment-1741702411 type _External = | 'discord' | 'github' | 'youtube' | 'twitter'; export type External = _External; type _Icons = | 'chevron-down' | 'download' | 'book-open' | 'link-external' | External | ModIcons; export type Icons = _Icons; // bypass for Astro compiler issue https://github.com/withastro/compiler/issues/554#issuecomment-1741702411 interface Props extends HTMLAttributes<'svg'> { icon: Icons path?: string size?: number | [number, number] } async function getSVG(name: string, path = 'impl') { const file = await import(`./${path}/${name}.svg?raw`); if (!file) throw new Error(`${name} not found`); const content = parse(file.default); const svg = content.querySelector('svg'); if (!svg) throw new Error(`${name} is not a valid SVG`); const { attributes, innerHTML } = svg; return { attributes, innerHTML, }; } const { icon, size, path = 'impl', ...attributes } = Astro.props as Props; let svgAttributes = {}; let html = ''; try { const sizeAttributes = () => { if (!size) return {}; if (Array.isArray(size)) { return { width: size[0], height: size[1], }; } return { width: size, height: size, }; }; const { attributes: baseAttributes, innerHTML } = await getSVG(icon, path); svgAttributes = { ...baseAttributes, ...attributes, ...sizeAttributes(), }; const colorRegex = /(fill|stroke)=\"([^"]*)\"/g; html = innerHTML.replaceAll(colorRegex, '$1="currentColor"'); } catch (err) { // ignored } ---