From f52f20e02f6908411ad9cddb341583456a3c2a8c Mon Sep 17 00:00:00 2001 From: Evan You Date: Mon, 14 Jun 2021 22:25:26 -0400 Subject: [PATCH] fix: avoid resetting head tags on hmr/page switch --- src/client/app/composables/head.ts | 46 +++++++++++++++++++++++------- 1 file changed, 36 insertions(+), 10 deletions(-) diff --git a/src/client/app/composables/head.ts b/src/client/app/composables/head.ts index 8fde0403ce82..6d1de07532a7 100644 --- a/src/client/app/composables/head.ts +++ b/src/client/app/composables/head.ts @@ -3,9 +3,9 @@ import { HeadConfig, SiteData } from '../../shared' import { Route } from '../router' export function useUpdateHead(route: Route, siteDataByRouteRef: Ref) { - const metaTags: HTMLElement[] = Array.from(document.querySelectorAll('meta')) - + let metaTagEls: HTMLElement[] = Array.from(document.querySelectorAll('meta')) let isFirstUpdate = true + const updateHeadTags = (newTags: HeadConfig[]) => { if (import.meta.env.PROD && isFirstUpdate) { // in production, the initial meta tags are already pre-rendered so we @@ -13,15 +13,41 @@ export function useUpdateHead(route: Route, siteDataByRouteRef: Ref) { isFirstUpdate = false return } - metaTags.forEach((el) => document.head.removeChild(el)) - metaTags.length = 0 - if (newTags && newTags.length) { - newTags.forEach((headConfig) => { - const el = createHeadElement(headConfig) - document.head.appendChild(el) - metaTags.push(el) - }) + + const newEls: HTMLElement[] = [] + const commonLength = Math.min(metaTagEls.length, newTags.length) + for (let i = 0; i < commonLength; i++) { + let el = metaTagEls[i] + const [tag, attrs] = newTags[i] + if (el.tagName.toLocaleLowerCase() === tag) { + for (const key in attrs) { + if (el.getAttribute(key) !== attrs[key]) { + el.setAttribute(key, attrs[key]) + } + } + for (let i = 0; i < el.attributes.length; i++) { + const name = el.attributes[i].name + if (!(name in attrs)) { + el.removeAttribute(name) + } + } + } else { + document.head.removeChild(el) + el = createHeadElement(newTags[i]) + document.head.append(el) + } + newEls.push(el) } + + metaTagEls + .slice(commonLength) + .forEach((el) => document.head.removeChild(el)) + newTags.slice(commonLength).forEach((headConfig) => { + const el = createHeadElement(headConfig) + document.head.appendChild(el) + newEls.push(el) + }) + metaTagEls = newEls } watchEffect(() => {