chore (SW-834): Upgrade to Next 15 * wip: apply codemod and upgrade swc plugin * wip: design-system to react 19, fix issues from async (search)params * wip: fix remaining issues from codemod serverClient is now async because context use headers() getLang is now async because it uses headers() * Minor cleanup * Inline react-material-symbols package Package is seemingly not maintained any more and doesn't support React 19. This copies the package source into `design-system`, makes the necessary changes for 19 and export it for others to use. * Fix missing awaits * Disable modal exit animations Enabling modal exit animations via isExiting prop is causing modals to be rendered in "hidden" state and never unmount. Seems to be an issue with react-aria-components, see https://github.com/adobe/react-spectrum/issues/7563. Can probably be fixed by rewriting to a solution similar to https://react-spectrum.adobe.com/react-aria/examples/framer-modal-sheet.html * Remove unstable cache implementation and use in memory cache locally * Fix ref type in SelectFilter * Use cloneElement to add key prop to element Approved-by: Linus Flood
95 lines
2.5 KiB
TypeScript
95 lines
2.5 KiB
TypeScript
import { env } from "@/env/server"
|
|
import { serverClient } from "@/lib/trpc/server"
|
|
|
|
import type { AlternateURLs } from "next/dist/lib/metadata/types/alternative-urls-types"
|
|
|
|
import type {
|
|
ContentTypeParams,
|
|
LangParams,
|
|
PageArgs,
|
|
UIDParams,
|
|
} from "@/types/params"
|
|
import type { Lang } from "@/constants/languages"
|
|
|
|
export async function generateMetadata({
|
|
searchParams,
|
|
params,
|
|
}: PageArgs<
|
|
LangParams & ContentTypeParams & UIDParams,
|
|
{ subpage?: string; filterFromUrl?: string }
|
|
>) {
|
|
const { lang } = await params
|
|
const { subpage, filterFromUrl, ...otherSearchParams } = await searchParams
|
|
// If there are other (real) search params, we don't want to index the page as this will
|
|
// cause duplicate content issues.
|
|
const noIndexOnSearchParams = !!Object.keys(otherSearchParams).length
|
|
const caller = await serverClient()
|
|
const { metadata, alternates } = await caller.contentstack.metadata.get({
|
|
subpage,
|
|
filterFromUrl,
|
|
noIndex: noIndexOnSearchParams,
|
|
})
|
|
|
|
if (!metadata) {
|
|
return {
|
|
robots: {
|
|
index: env.isLangLive(lang),
|
|
follow: env.isLangLive(lang),
|
|
},
|
|
}
|
|
}
|
|
|
|
if (typeof metadata?.robots === "string") {
|
|
return metadata
|
|
}
|
|
|
|
return {
|
|
...metadata,
|
|
robots: {
|
|
...(metadata.robots ?? {}),
|
|
index: isIndexable(metadata.robots?.index, lang, alternates),
|
|
follow: isIndexable(metadata.robots?.follow, lang, alternates),
|
|
},
|
|
}
|
|
}
|
|
|
|
function isIndexable(
|
|
pageIndexableFromSettings: boolean | null | undefined,
|
|
lang: Lang,
|
|
alternates: AlternateURLs | null
|
|
) {
|
|
// This is a special case for whitelisting the scandic friends pages, this can be removed when all pages are live
|
|
const url = getUrl(alternates)
|
|
const firstNonLangSegment = (url ?? "").substring(3)
|
|
if (firstNonLangSegment.startsWith("/scandic-friends")) {
|
|
return true
|
|
}
|
|
|
|
// If we are live we want to index the page, but if the page has been marked as noindex in contentstack we don't
|
|
return (pageIndexableFromSettings ?? true) && env.isLangLive(lang)
|
|
}
|
|
|
|
function getUrl(alternates: AlternateURLs | null): string | null {
|
|
try {
|
|
if (!alternates?.canonical) {
|
|
return null
|
|
}
|
|
|
|
if (typeof alternates.canonical === "string") {
|
|
return alternates.canonical
|
|
}
|
|
|
|
if ("href" in alternates.canonical) {
|
|
return alternates.canonical.href
|
|
}
|
|
|
|
if (typeof alternates.canonical.url === "string") {
|
|
return alternates.canonical.url
|
|
}
|
|
|
|
return alternates.canonical.url.href
|
|
} catch {
|
|
return null
|
|
}
|
|
}
|