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