@tanstack/vue-router
Version:
Modern and scalable routing for Vue applications
143 lines • 4.99 kB
JSX
import * as Vue from 'vue';
import { appendUniqueUserTags, escapeHtml, getAssetCrossOrigin, getScriptPreloadAttrs, resolveManifestCssLink, } from '@tanstack/router-core';
import { useStore } from '@tanstack/vue-store';
import { useRouter } from './useRouter';
export const useTags = (assetCrossOrigin) => {
const router = useRouter();
const matches = useStore(router.stores.matches, (value) => value);
const meta = Vue.computed(() => {
const resultMeta = [];
const metaByAttribute = {};
let title;
[...matches.value.map((match) => match.meta).filter(Boolean)]
.reverse()
.forEach((metas) => {
;
[...metas].reverse().forEach((m) => {
if (!m)
return;
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]) {
return;
}
else {
metaByAttribute[attribute] = true;
}
}
resultMeta.push({
tag: 'meta',
attrs: {
...m,
},
});
}
});
});
if (title) {
resultMeta.push(title);
}
resultMeta.reverse();
return resultMeta;
});
const links = Vue.computed(() => matches.value
.map((match) => match.links)
.filter(Boolean)
.flat(1)
.map((link) => ({
tag: 'link',
attrs: {
...link,
},
})));
const preloadMeta = Vue.computed(() => {
const preloadMeta = [];
matches.value.forEach((match) => {
router.ssr?.manifest?.routes[match.routeId]?.preloads
?.filter(Boolean)
.forEach((preload) => {
preloadMeta.push({
tag: 'link',
attrs: {
...getScriptPreloadAttrs(router.ssr?.manifest, preload, assetCrossOrigin),
},
});
});
});
return preloadMeta;
});
const headScripts = Vue.computed(() => matches.value
.map((match) => match.headScripts)
.flat(1)
.filter(Boolean).map(({ children, ...script }) => ({
tag: 'script',
attrs: {
...script,
},
children,
})));
const manifestAssets = Vue.computed(() => {
const manifest = router.ssr?.manifest;
const assets = [];
matches.value.forEach((match) => {
const routeManifest = manifest?.routes[match.routeId];
routeManifest?.css?.forEach((link) => {
const resolvedLink = resolveManifestCssLink(link);
assets.push({
tag: 'link',
attrs: {
rel: 'stylesheet',
...resolvedLink,
crossOrigin: getAssetCrossOrigin(assetCrossOrigin, 'stylesheet') ??
resolvedLink.crossOrigin,
},
});
});
});
if (manifest?.inlineStyle) {
assets.push({
tag: 'style',
attrs: manifest.inlineStyle.attrs,
children: manifest.inlineStyle.children,
inlineCss: true,
});
}
return assets;
});
return () => {
const tags = [];
tags.push(...manifestAssets.value);
appendUniqueUserTags(tags, meta.value);
tags.push(...preloadMeta.value);
appendUniqueUserTags(tags, links.value);
appendUniqueUserTags(tags, headScripts.value);
return tags;
};
};
//# sourceMappingURL=headContentUtils.jsx.map