import type { Awaitable } from '@antfu/utils'; import type { IconifyLoaderOptions } from './types'; const svgWidthRegex = /width\s*=\s*["'](\w+)["']/; const svgHeightRegex = /height\s*=\s*["'](\w+)["']/; function configureSvgSize( svg: string, props: Record, scale?: number ): [boolean, boolean] { const svgNode = svg.slice(0, svg.indexOf('>')); let result = svgWidthRegex.exec(svgNode); const w = result != null; if (typeof props.width === 'undefined' || props.width === null) { if (typeof scale === 'number') { props.width = `${scale}em`; } else if (result) { props.width = result[1]; } } result = svgHeightRegex.exec(svgNode); const h = result != null; if (typeof props.height === 'undefined' || props.height === null) { if (typeof scale === 'number') { props.height = `${scale}em`; } else if (result) { props.height = result[1]; } } return [w, h]; } export async function mergeIconProps( svg: string, collection: string, icon: string, options?: IconifyLoaderOptions, propsProvider?: () => Awaitable> ): Promise { const { scale, addXmlNs = false } = options ?? {}; const { additionalProps = {}, iconCustomizer } = options?.customizations ?? {}; const props: Record = (await propsProvider?.()) ?? {}; await iconCustomizer?.(collection, icon, props); Object.keys(additionalProps).forEach((p) => { const v = additionalProps[p]; if (v !== undefined && v !== null) props[p] = v; }); const [widthOnSvg, heightOnSvg] = configureSvgSize(svg, props, scale); // add xml namespaces if necessary if (addXmlNs) { // add svg xmlns if missing if (!svg.includes(' xmlns=') && !props['xmlns']) { props['xmlns'] = 'http://www.w3.org/2000/svg'; } // add xmlns:xlink if xlink present and the xmlns missing if ( !svg.includes(' xmlns:xlink=') && svg.includes('xlink:') && !props['xmlns:xlink'] ) { props['xmlns:xlink'] = 'http://www.w3.org/1999/xlink'; } } const propsToAdd = Object.keys(props) .map((p) => (p === 'width' && widthOnSvg) || (p === 'height' && heightOnSvg) ? null : `${p}="${props[p]}"` ) .filter((p) => p != null); if (propsToAdd.length) { svg = svg.replace(' { const v = props[p]; if (v !== undefined && v !== null) usedProps[p] = v; }); if (typeof props.width !== 'undefined' && props.width !== null) { usedProps.width = props.width; } if (typeof props.height !== 'undefined' && props.height !== null) { usedProps.height = props.height; } } return svg; }