|
@@ -162,7 +162,7 @@ function resolveComponents (route) {
|
|
|
})
|
|
|
}
|
|
|
|
|
|
-function callMiddleware (Components, context, layout) {
|
|
|
+function callMiddleware (Components, context, layout, renderState) {
|
|
|
let midd = ["redirect"]
|
|
|
let unknownMiddleware = false
|
|
|
|
|
@@ -194,10 +194,10 @@ function callMiddleware (Components, context, layout) {
|
|
|
if (unknownMiddleware) {
|
|
|
return
|
|
|
}
|
|
|
- return middlewareSeries(midd, context)
|
|
|
+ return middlewareSeries(midd, context, renderState)
|
|
|
}
|
|
|
|
|
|
-async function render (to, from, next) {
|
|
|
+async function render (to, from, next, renderState) {
|
|
|
if (this._routeChanged === false && this._paramChanged === false && this._queryChanged === false) {
|
|
|
return next()
|
|
|
}
|
|
@@ -228,6 +228,12 @@ async function render (to, from, next) {
|
|
|
await setContext(app, {
|
|
|
route: to,
|
|
|
from,
|
|
|
+ error: (err) => {
|
|
|
+ if (renderState.aborted) {
|
|
|
+ return
|
|
|
+ }
|
|
|
+ app.nuxt.error.call(this, err)
|
|
|
+ },
|
|
|
next: _next.bind(this)
|
|
|
})
|
|
|
this._dateLastError = app.nuxt.dateErr
|
|
@@ -240,10 +246,14 @@ async function render (to, from, next) {
|
|
|
// If no Components matched, generate 404
|
|
|
if (!Components.length) {
|
|
|
// Default layout
|
|
|
- await callMiddleware.call(this, Components, app.context)
|
|
|
+ await callMiddleware.call(this, Components, app.context, undefined, renderState)
|
|
|
if (nextCalled) {
|
|
|
return
|
|
|
}
|
|
|
+ if (renderState.aborted) {
|
|
|
+ next(false)
|
|
|
+ return
|
|
|
+ }
|
|
|
|
|
|
// Load layout for error page
|
|
|
const errorLayout = (NuxtError.options || NuxtError).layout
|
|
@@ -253,10 +263,14 @@ async function render (to, from, next) {
|
|
|
: errorLayout
|
|
|
)
|
|
|
|
|
|
- await callMiddleware.call(this, Components, app.context, layout)
|
|
|
+ await callMiddleware.call(this, Components, app.context, layout, renderState)
|
|
|
if (nextCalled) {
|
|
|
return
|
|
|
}
|
|
|
+ if (renderState.aborted) {
|
|
|
+ next(false)
|
|
|
+ return
|
|
|
+ }
|
|
|
|
|
|
// Show error page
|
|
|
app.context.error({ statusCode: 404, message: 'This page could not be found' })
|
|
@@ -276,10 +290,14 @@ async function render (to, from, next) {
|
|
|
|
|
|
try {
|
|
|
// Call middleware
|
|
|
- await callMiddleware.call(this, Components, app.context)
|
|
|
+ await callMiddleware.call(this, Components, app.context, undefined, renderState)
|
|
|
if (nextCalled) {
|
|
|
return
|
|
|
}
|
|
|
+ if (renderState.aborted) {
|
|
|
+ next(false)
|
|
|
+ return
|
|
|
+ }
|
|
|
if (app.context._errored) {
|
|
|
return next()
|
|
|
}
|
|
@@ -292,10 +310,14 @@ async function render (to, from, next) {
|
|
|
layout = await this.loadLayout(layout)
|
|
|
|
|
|
// Call middleware for layout
|
|
|
- await callMiddleware.call(this, Components, app.context, layout)
|
|
|
+ await callMiddleware.call(this, Components, app.context, layout, renderState)
|
|
|
if (nextCalled) {
|
|
|
return
|
|
|
}
|
|
|
+ if (renderState.aborted) {
|
|
|
+ next(false)
|
|
|
+ return
|
|
|
+ }
|
|
|
if (app.context._errored) {
|
|
|
return next()
|
|
|
}
|
|
@@ -400,9 +422,17 @@ async function render (to, from, next) {
|
|
|
|
|
|
// If not redirected
|
|
|
if (!nextCalled) {
|
|
|
+ if (renderState.aborted) {
|
|
|
+ next(false)
|
|
|
+ return
|
|
|
+ }
|
|
|
next()
|
|
|
}
|
|
|
} catch (err) {
|
|
|
+ if (renderState.aborted) {
|
|
|
+ next(false)
|
|
|
+ return
|
|
|
+ }
|
|
|
const error = err || {}
|
|
|
if (error.message === 'ERR_REDIRECT') {
|
|
|
return this.$nuxt.$emit('routeChanged', to, from, error)
|
|
@@ -437,7 +467,8 @@ function normalizeComponents (to, ___) {
|
|
|
})
|
|
|
}
|
|
|
|
|
|
-function setLayoutForNextPage (to) {
|
|
|
+const routeMap = new WeakMap()
|
|
|
+function getLayoutForNextPage (to, from, next) {
|
|
|
// Set layout
|
|
|
let hasError = Boolean(this.$options.nuxt.err)
|
|
|
if (this._hadError && this._dateLastError === this.$options.nuxt.dateErr) {
|
|
@@ -451,6 +482,21 @@ function setLayoutForNextPage (to) {
|
|
|
layout = layout(app.context)
|
|
|
}
|
|
|
|
|
|
+ routeMap.set(to, layout);
|
|
|
+
|
|
|
+ if (next) next();
|
|
|
+}
|
|
|
+
|
|
|
+function setLayoutForNextPage(to) {
|
|
|
+ const layout = routeMap.get(to)
|
|
|
+ routeMap.delete(to)
|
|
|
+
|
|
|
+ const prevPageIsError = this._hadError && this._dateLastError === this.$options.nuxt.dateErr
|
|
|
+
|
|
|
+ if (prevPageIsError) {
|
|
|
+ this.$options.nuxt.err = null
|
|
|
+ }
|
|
|
+
|
|
|
this.setLayout(layout)
|
|
|
}
|
|
|
|
|
@@ -543,6 +589,7 @@ async function mountApp (__app) {
|
|
|
// Add afterEach router hooks
|
|
|
router.afterEach(normalizeComponents)
|
|
|
|
|
|
+ router.beforeResolve(getLayoutForNextPage.bind(_app))
|
|
|
router.afterEach(setLayoutForNextPage.bind(_app))
|
|
|
|
|
|
router.afterEach(fixPrepatch.bind(_app))
|
|
@@ -568,11 +615,22 @@ async function mountApp (__app) {
|
|
|
_app.$loading = {} // To avoid error while _app.$nuxt does not exist
|
|
|
if (NUXT.error) {
|
|
|
_app.error(NUXT.error)
|
|
|
+ _app.nuxt.errPageReady = true
|
|
|
}
|
|
|
|
|
|
// Add beforeEach router hooks
|
|
|
router.beforeEach(loadAsyncComponents.bind(_app))
|
|
|
- router.beforeEach(render.bind(_app))
|
|
|
+
|
|
|
+ // Each new invocation of render() aborts previous invocation
|
|
|
+ let renderState = null
|
|
|
+ const boundRender = render.bind(_app)
|
|
|
+ router.beforeEach((to, from, next) => {
|
|
|
+ if (renderState) {
|
|
|
+ renderState.aborted = true
|
|
|
+ }
|
|
|
+ renderState = { aborted: false }
|
|
|
+ boundRender(to, from, next, renderState)
|
|
|
+ })
|
|
|
|
|
|
// Fix in static: remove trailing slash to force hydration
|
|
|
// Full static, if server-rendered: hydrate, to allow custom redirect to generated page
|
|
@@ -582,10 +640,15 @@ async function mountApp (__app) {
|
|
|
return mount()
|
|
|
}
|
|
|
|
|
|
+ const clientFirstLayoutSet = () => {
|
|
|
+ getLayoutForNextPage.call(_app, router.currentRoute)
|
|
|
+ setLayoutForNextPage.call(_app, router.currentRoute)
|
|
|
+ }
|
|
|
+
|
|
|
// First render on client-side
|
|
|
const clientFirstMount = () => {
|
|
|
normalizeComponents(router.currentRoute, router.currentRoute)
|
|
|
- setLayoutForNextPage.call(_app, router.currentRoute)
|
|
|
+ clientFirstLayoutSet()
|
|
|
checkForErrors(_app)
|
|
|
// Don't call fixPrepatch.call(_app, router.currentRoute, router.currentRoute) since it's first render
|
|
|
mount()
|
|
@@ -613,5 +676,6 @@ async function mountApp (__app) {
|
|
|
errorHandler(err)
|
|
|
}
|
|
|
})
|
|
|
- })
|
|
|
+ },
|
|
|
+ { aborted: false })
|
|
|
}
|