mirror of
https://github.com/nuxt/nuxt.git
synced 2025-02-11 11:18:06 +00:00
246 lines
8.3 KiB
TypeScript
246 lines
8.3 KiB
TypeScript
import { getCurrentInstance, hasInjectionContext, inject, onUnmounted } from 'vue'
|
|
import type { Ref } from 'vue'
|
|
import type { NavigationFailure, NavigationGuard, RouteLocationNormalized, RouteLocationPathRaw, RouteLocationRaw, Router, useRoute as _useRoute, useRouter as _useRouter } from '#vue-router'
|
|
import { sanitizeStatusCode } from 'h3'
|
|
import { hasProtocol, isScriptProtocol, joinURL, parseURL, withQuery } from 'ufo'
|
|
|
|
import { useNuxtApp, useRuntimeConfig } from '../nuxt'
|
|
import type { NuxtError } from './error'
|
|
import { createError, showError } from './error'
|
|
import { useState } from './state'
|
|
|
|
import type { PageMeta } from '#app'
|
|
import { PageRouteSymbol } from '#app/components/injections'
|
|
|
|
export const useRouter: typeof _useRouter = () => {
|
|
return useNuxtApp()?.$router as Router
|
|
}
|
|
|
|
export const useRoute: typeof _useRoute = () => {
|
|
if (process.dev && isProcessingMiddleware()) {
|
|
console.warn('[nuxt] Calling `useRoute` within middleware may lead to misleading results. Instead, use the (to, from) arguments passed to the middleware to access the new and old routes.')
|
|
}
|
|
if (hasInjectionContext()) {
|
|
return inject(PageRouteSymbol, useNuxtApp()._route)
|
|
}
|
|
return useNuxtApp()._route
|
|
}
|
|
|
|
export const onBeforeRouteLeave = (guard: NavigationGuard) => {
|
|
const unsubscribe = useRouter().beforeEach((to, from, next) => {
|
|
if (to === from) { return }
|
|
return guard(to, from, next)
|
|
})
|
|
onUnmounted(unsubscribe)
|
|
}
|
|
|
|
export const onBeforeRouteUpdate = (guard: NavigationGuard) => {
|
|
const unsubscribe = useRouter().beforeEach(guard)
|
|
onUnmounted(unsubscribe)
|
|
}
|
|
|
|
export interface RouteMiddleware {
|
|
(to: RouteLocationNormalized, from: RouteLocationNormalized): ReturnType<NavigationGuard>
|
|
}
|
|
|
|
/*! @__NO_SIDE_EFFECTS__ */
|
|
export function defineNuxtRouteMiddleware (middleware: RouteMiddleware) {
|
|
return middleware
|
|
}
|
|
|
|
export interface AddRouteMiddlewareOptions {
|
|
global?: boolean
|
|
}
|
|
|
|
interface AddRouteMiddleware {
|
|
(name: string, middleware: RouteMiddleware, options?: AddRouteMiddlewareOptions): void
|
|
(middleware: RouteMiddleware): void
|
|
}
|
|
|
|
export const addRouteMiddleware: AddRouteMiddleware = (name: string | RouteMiddleware, middleware?: RouteMiddleware, options: AddRouteMiddlewareOptions = {}) => {
|
|
const nuxtApp = useNuxtApp()
|
|
const global = options.global || typeof name !== 'string'
|
|
const mw = typeof name !== 'string' ? name : middleware
|
|
if (!mw) {
|
|
console.warn('[nuxt] No route middleware passed to `addRouteMiddleware`.', name)
|
|
return
|
|
}
|
|
if (global) {
|
|
nuxtApp._middleware.global.push(mw)
|
|
} else {
|
|
nuxtApp._middleware.named[name] = mw
|
|
}
|
|
}
|
|
|
|
const isProcessingMiddleware = () => {
|
|
try {
|
|
if (useNuxtApp()._processingMiddleware) {
|
|
return true
|
|
}
|
|
} catch {
|
|
// Within an async middleware
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Conditional types, either one or other
|
|
type Without<T, U> = { [P in Exclude<keyof T, keyof U>]?: never }
|
|
type XOR<T, U> = (T | U) extends Object ? (Without<T, U> & U) | (Without<U, T> & T) : T | U
|
|
|
|
export type OpenWindowFeatures = {
|
|
popup?: boolean
|
|
noopener?: boolean
|
|
noreferrer?: boolean
|
|
} & XOR<{width?: number}, {innerWidth?: number}>
|
|
& XOR<{height?: number}, {innerHeight?: number}>
|
|
& XOR<{left?: number}, {screenX?: number}>
|
|
& XOR<{top?: number}, {screenY?: number}>
|
|
|
|
export type OpenOptions = {
|
|
target: '_blank' | '_parent' | '_self' | '_top' | (string & {})
|
|
windowFeatures?: OpenWindowFeatures
|
|
}
|
|
|
|
export interface NavigateToOptions {
|
|
replace?: boolean
|
|
redirectCode?: number
|
|
external?: boolean
|
|
open?: OpenOptions
|
|
}
|
|
|
|
export const navigateTo = (to: RouteLocationRaw | undefined | null, options?: NavigateToOptions): Promise<void | NavigationFailure | false> | false | void | RouteLocationRaw => {
|
|
if (!to) {
|
|
to = '/'
|
|
}
|
|
|
|
const toPath = typeof to === 'string' ? to : (withQuery((to as RouteLocationPathRaw).path || '/', to.query || {}) + (to.hash || ''))
|
|
|
|
// Early open handler
|
|
if (options?.open) {
|
|
if (process.client) {
|
|
const { target = '_blank', windowFeatures = {} } = options.open
|
|
|
|
const features = Object.entries(windowFeatures)
|
|
.filter(([_, value]) => value !== undefined)
|
|
.map(([feature, value]) => `${feature.toLowerCase()}=${value}`)
|
|
.join(', ')
|
|
|
|
open(toPath, target, features)
|
|
}
|
|
|
|
return Promise.resolve()
|
|
}
|
|
|
|
const isExternal = options?.external || hasProtocol(toPath, { acceptRelative: true })
|
|
if (isExternal) {
|
|
if (!options?.external) {
|
|
throw new Error('Navigating to an external URL is not allowed by default. Use `navigateTo(url, { external: true })`.')
|
|
}
|
|
const protocol = parseURL(toPath).protocol
|
|
if (protocol && isScriptProtocol(protocol)) {
|
|
throw new Error(`Cannot navigate to a URL with '${protocol}' protocol.`)
|
|
}
|
|
}
|
|
|
|
const inMiddleware = isProcessingMiddleware()
|
|
|
|
// Early redirect on client-side
|
|
if (process.client && !isExternal && inMiddleware) {
|
|
return to
|
|
}
|
|
|
|
const router = useRouter()
|
|
|
|
const nuxtApp = useNuxtApp()
|
|
|
|
if (process.server) {
|
|
if (nuxtApp.ssrContext) {
|
|
const fullPath = typeof to === 'string' || isExternal ? toPath : router.resolve(to).fullPath || '/'
|
|
const location = isExternal ? toPath : joinURL(useRuntimeConfig().app.baseURL, fullPath)
|
|
|
|
async function redirect (response: any) {
|
|
// TODO: consider deprecating in favour of `app:rendered` and removing
|
|
await nuxtApp.callHook('app:redirected')
|
|
const encodedLoc = location.replace(/"/g, '%22')
|
|
nuxtApp.ssrContext!._renderResponse = {
|
|
statusCode: sanitizeStatusCode(options?.redirectCode || 302, 302),
|
|
body: `<!DOCTYPE html><html><head><meta http-equiv="refresh" content="0; url=${encodedLoc}"></head></html>`,
|
|
headers: { location }
|
|
}
|
|
return response
|
|
}
|
|
|
|
// We wait to perform the redirect last in case any other middleware will intercept the redirect
|
|
// and redirect somewhere else instead.
|
|
if (!isExternal && inMiddleware) {
|
|
router.afterEach(final => final.fullPath === fullPath ? redirect(false) : undefined)
|
|
return to
|
|
}
|
|
return redirect(!inMiddleware ? undefined : /* abort route navigation */ false)
|
|
}
|
|
}
|
|
|
|
// Client-side redirection using vue-router
|
|
if (isExternal) {
|
|
if (options?.replace) {
|
|
location.replace(toPath)
|
|
} else {
|
|
location.href = toPath
|
|
}
|
|
// Within in a Nuxt route middleware handler
|
|
if (inMiddleware) {
|
|
// Abort navigation when app is hydrated
|
|
if (!nuxtApp.isHydrating) {
|
|
return false
|
|
}
|
|
// When app is hydrating (i.e. on page load), we don't want to abort navigation as
|
|
// it would lead to a 404 error / page that's blinking before location changes.
|
|
return new Promise(() => {})
|
|
}
|
|
return Promise.resolve()
|
|
}
|
|
|
|
return options?.replace ? router.replace(to) : router.push(to)
|
|
}
|
|
|
|
/** This will abort navigation within a Nuxt route middleware handler. */
|
|
export const abortNavigation = (err?: string | Partial<NuxtError>) => {
|
|
if (process.dev && !isProcessingMiddleware()) {
|
|
throw new Error('abortNavigation() is only usable inside a route middleware handler.')
|
|
}
|
|
|
|
if (!err) { return false }
|
|
|
|
err = createError(err)
|
|
|
|
if (err.fatal) {
|
|
useNuxtApp().runWithContext(() => showError(err as NuxtError))
|
|
}
|
|
|
|
throw err
|
|
}
|
|
|
|
export const setPageLayout = (layout: unknown extends PageMeta['layout'] ? string : PageMeta['layout']) => {
|
|
if (process.server) {
|
|
if (process.dev && getCurrentInstance() && useState('_layout').value !== layout) {
|
|
console.warn('[warn] [nuxt] `setPageLayout` should not be called to change the layout on the server within a component as this will cause hydration errors.')
|
|
}
|
|
useState('_layout').value = layout
|
|
}
|
|
const nuxtApp = useNuxtApp()
|
|
if (process.dev && nuxtApp.isHydrating && nuxtApp.payload.serverRendered && useState('_layout').value !== layout) {
|
|
console.warn('[warn] [nuxt] `setPageLayout` should not be called to change the layout during hydration as this will cause hydration errors.')
|
|
}
|
|
const inMiddleware = isProcessingMiddleware()
|
|
if (inMiddleware || process.server || nuxtApp.isHydrating) {
|
|
const unsubscribe = useRouter().beforeResolve((to) => {
|
|
to.meta.layout = layout as Exclude<PageMeta['layout'], Ref | false>
|
|
unsubscribe()
|
|
})
|
|
}
|
|
if (!inMiddleware) {
|
|
useRoute().meta.layout = layout as Exclude<PageMeta['layout'], Ref | false>
|
|
}
|
|
}
|