Nuxt/packages/nuxt3/src/app/nuxt.ts

202 lines
5.2 KiB
TypeScript
Raw Normal View History

/* eslint-disable no-use-before-define */
import { getCurrentInstance, reactive } from 'vue'
import type { App, VNode } from 'vue'
import { createHooks, Hookable } from 'hookable'
import { legacyPlugin, LegacyContext } from './legacy'
type NuxtMeta = {
htmlAttrs?: string
headAttrs?: string
bodyAttrs?: string
headTags?: string
bodyPrepend?: string
bodyScripts?: string
}
type HookResult = Promise<void> | void
export interface RuntimeNuxtHooks {
'app:created': (app: App<Element>) => HookResult
'app:beforeMount': (app: App<Element>) => HookResult
'app:mounted': (app: App<Element>) => HookResult
'app:rendered': () => HookResult
'page:start': (Component?: VNode) => HookResult
'page:finish': (Component?: VNode) => HookResult
'meta:register': (metaRenderers: Array<(nuxt: NuxtApp) => NuxtMeta | Promise<NuxtMeta>>) => HookResult
}
export interface NuxtApp {
vueApp: App<Element>
globalName: string
hooks: Hookable<RuntimeNuxtHooks>
hook: NuxtApp['hooks']['hook']
callHook: NuxtApp['hooks']['callHook']
[key: string]: any
_asyncDataPromises?: Record<string, Promise<any>>
_legacyContext?: LegacyContext
ssrContext?: Record<string, any> & {
renderMeta?: () => Promise<NuxtMeta> | NuxtMeta
}
payload: {
2021-03-17 09:17:18 +00:00
serverRendered?: true
data?: Record<string, any>
2021-10-11 17:48:03 +00:00
state?: Record<string, any>
rendered?: Function
[key: string]: any
}
provide: (name: string, value: any) => void
}
export const NuxtPluginIndicator = '__nuxt_plugin'
export interface Plugin {
(nuxt: NuxtApp): Promise<void> | void
[NuxtPluginIndicator]?: true
}
export interface LegacyPlugin {
(context: LegacyContext, provide: NuxtApp['provide']): Promise<void> | void
}
export interface CreateOptions {
vueApp: NuxtApp['vueApp']
ssrContext?: NuxtApp['ssrContext']
globalName?: NuxtApp['globalName']
}
export function createNuxtApp (options: CreateOptions) {
const nuxtApp: NuxtApp = {
provide: undefined,
globalName: 'nuxt',
2021-10-11 17:48:03 +00:00
payload: reactive({
data: {},
state: {},
...(process.client ? window.__NUXT__ : { serverRendered: true })
}),
isHydrating: process.client,
_asyncDataPromises: {},
...options
} as any as NuxtApp
nuxtApp.hooks = createHooks<RuntimeNuxtHooks>()
nuxtApp.hook = nuxtApp.hooks.hook
nuxtApp.callHook = nuxtApp.hooks.callHook
nuxtApp.provide = (name: string, value: any) => {
const $name = '$' + name
defineGetter(nuxtApp, $name, value)
defineGetter(nuxtApp.vueApp.config.globalProperties, $name, value)
}
// Inject $nuxt
defineGetter(nuxtApp.vueApp, '$nuxt', nuxtApp)
defineGetter(nuxtApp.vueApp.config.globalProperties, '$nuxt', nuxtApp)
// Expose nuxt to the renderContext
if (nuxtApp.ssrContext) {
nuxtApp.ssrContext.nuxt = nuxtApp
}
if (process.server) {
// Expose to server renderer to create window.__NUXT__
nuxtApp.ssrContext = nuxtApp.ssrContext || {}
nuxtApp.ssrContext.payload = nuxtApp.payload
}
// Expose runtime config
if (process.server) {
nuxtApp.provide('config', options.ssrContext.runtimeConfig.private)
nuxtApp.payload.config = options.ssrContext.runtimeConfig.public
} else {
nuxtApp.provide('config', reactive(nuxtApp.payload.config))
}
return nuxtApp
}
export function applyPlugin (nuxtApp: NuxtApp, plugin: Plugin) {
if (typeof plugin !== 'function') { return }
return callWithNuxt(nuxtApp, () => plugin(nuxtApp))
}
export async function applyPlugins (nuxtApp: NuxtApp, plugins: Plugin[]) {
for (const plugin of plugins) {
await applyPlugin(nuxtApp, plugin)
}
}
2021-04-09 13:48:39 +00:00
export function normalizePlugins (_plugins: Array<Plugin | LegacyPlugin>) {
let needsLegacyContext = false
const plugins = _plugins.map((plugin) => {
if (typeof plugin !== 'function') {
return () => {}
}
if (isLegacyPlugin(plugin)) {
needsLegacyContext = true
return (nuxtApp: NuxtApp) => plugin(nuxtApp._legacyContext!, nuxtApp.provide)
}
return plugin
})
if (needsLegacyContext) {
plugins.unshift(legacyPlugin)
}
return plugins as Plugin[]
}
export function defineNuxtPlugin (plugin: Plugin) {
plugin[NuxtPluginIndicator] = true
return plugin
}
export function isLegacyPlugin (plugin: unknown): plugin is LegacyPlugin {
return !plugin[NuxtPluginIndicator]
}
2021-10-11 17:48:03 +00:00
let currentNuxtAppInstance: NuxtApp | null
2021-04-09 13:48:39 +00:00
2021-10-11 17:48:03 +00:00
export const setNuxtAppInstance = (nuxt: NuxtApp | null) => {
currentNuxtAppInstance = nuxt
2021-04-09 13:48:39 +00:00
}
/**
* Ensures that the setup function passed in has access to the Nuxt instance via `useNuxt`.
2021-04-15 18:49:29 +00:00
*
2021-04-09 13:48:39 +00:00
* @param nuxt A Nuxt instance
* @param setup The function to call
*/
export async function callWithNuxt (nuxt: NuxtApp, setup: () => any) {
2021-10-11 17:48:03 +00:00
setNuxtAppInstance(nuxt)
2021-04-09 13:48:39 +00:00
const p = setup()
2021-10-11 17:48:03 +00:00
setNuxtAppInstance(null)
2021-04-09 13:48:39 +00:00
await p
}
/**
* Returns the current Nuxt instance.
*/
export function useNuxtApp (): NuxtApp {
2021-04-09 13:48:39 +00:00
const vm = getCurrentInstance()
if (!vm) {
2021-10-11 17:48:03 +00:00
if (!currentNuxtAppInstance) {
2021-04-09 13:48:39 +00:00
throw new Error('nuxt instance unavailable')
}
2021-10-11 17:48:03 +00:00
return currentNuxtAppInstance
2021-04-09 13:48:39 +00:00
}
return vm.appContext.app.$nuxt
}
export function useRuntimeConfig (): Record<string, any> {
return useNuxtApp().$config
}
function defineGetter<K extends string | number | symbol, V> (obj: Record<K, V>, key: K, val: V) {
Object.defineProperty(obj, key, { get: () => val })
}