UNPKG

@tanstack/solid-router

Version:

Modern and scalable routing for Solid applications

209 lines 7.15 kB
import * as Solid from 'solid-js'; import { appendUniqueUserTags, escapeHtml, getAssetCrossOrigin, getScriptPreloadAttrs, resolveManifestCssLink, } from '@tanstack/router-core'; import { useRouter } from './useRouter'; /** * Build the list of head/link/meta/script tags to render for active matches. * Used internally by `HeadContent`. */ export const useTags = (assetCrossOrigin) => { const router = useRouter(); const nonce = router.options.ssr?.nonce; const activeMatches = Solid.createMemo(() => router.stores.matches.get()); const routeMeta = Solid.createMemo(() => activeMatches() .map((match) => match.meta) .filter((meta) => meta !== undefined)); const meta = Solid.createMemo(() => { const resultMeta = []; const metaByAttribute = {}; let title; const routeMetasArray = routeMeta(); for (let i = routeMetasArray.length - 1; i >= 0; i--) { const metas = routeMetasArray[i]; for (let j = metas.length - 1; j >= 0; j--) { const m = metas[j]; if (!m) continue; if (m.title) { if (!title) { title = { tag: 'title', children: m.title, }; } } else if ('script:ld+json' in m) { // Handle JSON-LD structured data // Content is HTML-escaped to prevent XSS when injected via innerHTML try { const json = JSON.stringify(m['script:ld+json']); resultMeta.push({ tag: 'script', attrs: { type: 'application/ld+json', }, children: escapeHtml(json), }); } catch { // Skip invalid JSON-LD objects } } else { const attribute = m.name ?? m.property; if (attribute) { if (metaByAttribute[attribute]) { continue; } else { metaByAttribute[attribute] = true; } } resultMeta.push({ tag: 'meta', attrs: { ...m, nonce, }, }); } } } if (title) { resultMeta.push(title); } if (router.options.ssr?.nonce) { resultMeta.push({ tag: 'meta', attrs: { property: 'csp-nonce', content: router.options.ssr.nonce, }, }); } resultMeta.reverse(); return resultMeta; }); const links = Solid.createMemo(() => { const matches = activeMatches(); const constructed = matches .flatMap((match) => match.links ?? []) .filter((link) => link !== undefined) .map((link) => ({ tag: 'link', attrs: { ...link, nonce, }, })); return constructed; }); const manifestCssTags = Solid.createMemo(() => { const manifest = router.ssr?.manifest; const tags = []; if (!manifest) { return tags; } for (const match of activeMatches()) { manifest.routes[match.routeId]?.css?.forEach((link) => { const resolvedLink = resolveManifestCssLink(link); tags.push({ tag: 'link', attrs: { rel: 'stylesheet', ...resolvedLink, crossOrigin: getAssetCrossOrigin(assetCrossOrigin, 'stylesheet') ?? resolvedLink.crossOrigin, nonce, }, }); }); } if (manifest.inlineStyle) { tags.push({ tag: 'style', attrs: { ...manifest.inlineStyle.attrs, nonce, }, children: manifest.inlineStyle.children, inlineCss: true, }); } return tags; }); const preloadLinks = Solid.createMemo(() => { const matches = activeMatches(); const preloadLinks = []; matches.forEach((match) => router.ssr?.manifest?.routes[match.routeId]?.preloads ?.filter(Boolean) .forEach((preload) => { preloadLinks.push({ tag: 'link', attrs: { ...getScriptPreloadAttrs(router.ssr?.manifest, preload, assetCrossOrigin), nonce, }, }); })); return preloadLinks; }); const styles = Solid.createMemo(() => { return activeMatches() .flatMap((match) => match.styles ?? []) .filter((style) => style !== undefined) .map(({ children, ...style }) => ({ tag: 'style', attrs: { ...style, nonce, }, children: children, })); }); const headScripts = Solid.createMemo(() => { return activeMatches() .flatMap((match) => match.headScripts ?? []) .filter((script) => script !== undefined) .map(({ children, ...script }) => ({ tag: 'script', attrs: { ...script, nonce, }, children: children, })); }); return Solid.createMemo((prev) => { const next = []; appendUniqueUserTags(next, meta()); next.push(...preloadLinks()); appendUniqueUserTags(next, links()); next.push(...manifestCssTags()); appendUniqueUserTags(next, styles()); appendUniqueUserTags(next, headScripts()); if (prev === undefined) { return next; } return replaceEqualTags(prev, next); }); }; function replaceEqualTags(prev, next) { const prevByKey = new Map(); for (const tag of prev) { prevByKey.set(JSON.stringify(tag), tag); } let isEqual = prev.length === next.length; const result = next.map((tag, index) => { const existing = prevByKey.get(JSON.stringify(tag)); if (existing) { if (existing !== prev[index]) { isEqual = false; } return existing; } isEqual = false; return tag; }); return isEqual ? prev : result; } //# sourceMappingURL=headContentUtils.jsx.map