我正在使用 nuxt 生成完整的静态 Web 应用程序,如此处所述https://nuxtjs.org/blog/going-full-static/#crazy-fast-static-applications
我也有一个小博客作为静态站点加载,所以我使用 fetch 钩子从 api 加载数据。
async fetch() {
this.posts = await fetch(`${this.baseApi}/posts`).then(res => res.json())
},
当我生成( npm run generate
)时,获取的状态是在里面正确生成的dist/assets/static
,所以直接访问时/blog
,状态正确加载,数据正确显示。但是,当我在主页中并使用
this.$router.push
或一个
<nuxt-link to="/blog">Blog</nuxt-link>
获取的状态没有被加载,我必须再次调用api,或者在钩子中再调用this.$fetch()
一次mounted()
我已经添加了一个
watch: {
'$route.query': '$fetch'
}
到主页
我需要在使用导航时正确加载获取的状态我还缺少什么?
澄清
我没有遇到 fetch 钩子本身的任何问题,而是导航没有检索目标路由的状态。连HTML都在那里我需要页面来获取目标路由的状态,当路由发生变化时,因为vue模板依赖它,所以如果没有加载,ui就不会显示任何东西,我是被迫的手动调用 fetch 钩子
为了看得更清楚,这是我的 devtools 在直接访问 /blog 时的屏幕截图,注意 state.js 是如何正确检索的(它包含所有呈现的内容)
以下是我的 devtools 在访问 / 时的屏幕截图,然后使用 nuxt-link 或 this.$router.push 去博客(结果相同)
Blog.vue
<template>
<b-container class="container blog">
<b-row>
<b-col lg="12" md="12" sm="12" cols="12" class="logo-col">
<SbLogoSingle />
</b-col>
</b-row>
<b-row v-if="$fetchState.pending" class="text-center">
<b-spinner style="margin: auto"></b-spinner>
</b-row>
<b-row v-else>
<b-col
v-for="(post, idx) in posts.data"
:key="idx"
lg="4"
md="4"
sm="6"
cols="12"
class="blog-post-col"
>
<b-card
v-if="post !== undefined"
no-body
class="shadow-lg blog-post-card"
:img-src="post.media.url"
img-top
>
<b-card-body class="text-left">
<b-card-title>{{ replaceSlugByString(post.slug) }}</b-card-title>
<b-card-text
class="post-short-description"
v-html="post.localizations[0].shortDescription"
></b-card-text>
</b-card-body>
<template #footer>
<div class="text-left">
<b-button class="apply-btn read-more-btn" @click="openBlogPost(idx)">Read more</b-button>
</div>
</template>
</b-card>
</b-col>
</b-row>
</b-container>
</template>
<script>
import { mapState } from 'vuex'
export default {
data() {
return {
slug: 'test',
posts: {},
currentPage: 1,
perPage: 12,
pageIndex: 1,
totalPages: 1,
}
},
async fetch() {
const response = await fetch(`${this.baseApi}/StaticPage`)
const fetchedPosts = await response.json()
this.posts = fetchedPosts
// this.posts = await fetch(`${this.baseApi}/StaticPage`).then(res =>res.json())
},
computed: {
...mapState('modules/settings', ['baseApi']),
},
beforeMount() {
this.$fetch() // i want to remove this because the pages are statically generated correctly, I'm only adding it to refresh the state. which can be retrieved as a separate js file when accessing the route directly
},
methods: {
openBlogPost(idx) {
const pageObject = this.posts.data[idx]
this.$router.push({
name: `blog-slug`,
params: {
slug: pageObject.slug,
page: pageObject,
},
})
},
replaceSlugByString(slug) {
return slug.replaceAll('-', ' ')
},
},
}
</script>
这是 slug.vue 的 pastebin