import type { InjectionKey, Ref, VNode } from 'vue' import { Suspense, Transition, computed, defineComponent, h, inject, mergeProps, nextTick, onMounted, provide, ref, unref } from 'vue' import type { RouteLocationNormalizedLoaded } from 'vue-router' import { _wrapIf } from './utils' import { useRoute } from '#app/composables/router' // @ts-expect-error virtual file import { useRoute as useVueRouterRoute } from '#build/pages' // @ts-expect-error virtual file import layouts from '#build/layouts' // @ts-expect-error virtual file import { appLayoutTransition as defaultLayoutTransition } from '#build/nuxt.config.mjs' import { useNuxtApp } from '#app' export interface LayoutMeta { isCurrent: (route: RouteLocationNormalizedLoaded) => boolean } export const LayoutMetaSymbol: InjectionKey = Symbol('layout-meta') export default defineComponent({ name: 'NuxtLayout', inheritAttrs: false, props: { name: { type: [String, Boolean, Object] as unknown as () => string | false | Ref, default: null } }, setup (props, context) { const nuxtApp = useNuxtApp() // Need to ensure (if we are not a child of ``) that we use synchronous route (not deferred) const injectedRoute = inject('_route') as RouteLocationNormalizedLoaded const route = injectedRoute === useRoute() ? useVueRouterRoute() : injectedRoute const layout = computed(() => unref(props.name) ?? route.meta.layout as string ?? 'default') const layoutRef = ref() context.expose({ layoutRef }) return () => { const done = nuxtApp.deferHydration() const hasLayout = layout.value && layout.value in layouts if (process.dev && layout.value && !hasLayout && layout.value !== 'default') { console.warn(`Invalid layout \`${layout.value}\` selected.`) } const transitionProps = route.meta.layoutTransition ?? defaultLayoutTransition // We avoid rendering layout transition if there is no layout to render return _wrapIf(Transition, hasLayout && transitionProps, { default: () => h(Suspense, { suspensible: true, onResolve: () => { nextTick(done) } }, { default: () => _wrapIf(LayoutProvider, hasLayout && { layoutProps: mergeProps(context.attrs, { ref: layoutRef }), key: layout.value, name: layout.value, shouldProvide: !props.name, hasTransition: !!transitionProps }, context.slots).default() }) }).default() } } }) const LayoutProvider = defineComponent({ name: 'NuxtLayoutProvider', inheritAttrs: false, props: { name: { type: String }, layoutProps: { type: Object }, hasTransition: { type: Boolean }, shouldProvide: { type: Boolean } }, setup (props, context) { // Prevent reactivity when the page will be rerendered in a different suspense fork if (props.shouldProvide) { // eslint-disable-next-line vue/no-setup-props-destructure const name = props.name provide(LayoutMetaSymbol, { isCurrent: (route: RouteLocationNormalizedLoaded) => name === (route.meta.layout ?? 'default') }) } let vnode: VNode if (process.dev && process.client) { onMounted(() => { nextTick(() => { if (['#comment', '#text'].includes(vnode?.el?.nodeName)) { console.warn(`[nuxt] \`${props.name}\` layout does not have a single root node and will cause errors when navigating between routes.`) } }) }) } return () => { if (process.dev && process.client && props.hasTransition) { vnode = h(layouts[props.name], props.layoutProps, context.slots) return vnode } return h(layouts[props.name], props.layoutProps, context.slots) } } })