1. Nuxt.js服务器端渲染
学习目标
- 了解Nuxt.js的作用
- 掌握Nuxt.js中的路由
- 掌握layouts、pages以及components的区别
- 能够在Nuxt.js项目中使用element-ui
- 掌握Nuxt.js中异步获取数据的方式
- 完成豆瓣电影小案例
- 掌握SEO的优化
1.1 Nuxt.js入门
1.1.1 什么是Nuxt.js
Vue服务端渲染官网
Nuxt.js官网
Nuxt.js 是一个基于 Vue.js 的通用应用框架。
1.1.2 第一个Nuxt应用程序
npm i create-nuxt-app -g create-nuxt-app my-nuxt-demo cd my-nuxt-demo npm run dev
|
1.1.3 文件结构分析
└─my-nuxt-demo ├─.nuxt // Nuxt自动生成,临时的用于编辑的文件,build ├─assets // 用于组织未编译的静态资源如LESS、SASS或JavaScript,对于不需要通过 Webpack 处理的静态资源文件,可以放置在 static 目录中 ├─components // 用于自己编写的Vue组件,比如日历组件、分页组件 ├─layouts // 布局目录,用于组织应用的布局组件,不可更改⭐ ├─middleware // 用于存放中间件 ├─node_modules ├─pages // 用于组织应用的路由及视图,Nuxt.js根据该目录结构自动生成对应的路由配置,文件名不可更改⭐ ├─plugins // 用于组织那些需要在 根vue.js应用 实例化之前需要运行的 Javascript 插件。 ├─static // 用于存放应用的静态文件,此类文件不会被 Nuxt.js 调用 Webpack 进行构建编译处理。 服务器启动的时候,该目录下的文件会映射至应用的根路径 / 下。文件夹名不可更改。⭐ └─store // 用于组织应用的Vuex 状态管理。文件夹名不可更改。⭐ ├─.editorconfig // 开发工具格式配置 ├─.eslintrc.js // ESLint的配置文件,用于检查代码格式 ├─.gitignore // 配置git忽略文件 ├─nuxt.config.js // 用于组织Nuxt.js 应用的个性化配置,以便覆盖默认配置。文件名不可更改。⭐ ├─package-lock.json // npm自动生成,用于帮助package的统一设置的,yarn也有相同的操作 ├─package.json // npm 包管理配置文件 ├─README.md
|
1.2 页面和路由
1.2.1 基本路由
Nuxt.js 依据 pages
目录结构自动生成 vue-router 模块的路由配置。
假设 pages
的目录结构如下
└─pages ├─index.vue └─user ├─index.vue ├─one.vue
|
那么,Nuxt.js 自动生成的路由配置如下:
router: { routes: [ { name: 'index', path: '/', component: 'pages/index.vue' }, { name: 'user', path: '/user', component: 'pages/user/index.vue' }, { name: 'user-one', path: '/user/one', component: 'pages/user/one.vue' } ] }
|
1.2.2 页面跳转
- 不要写成a标签,因为是重新获取一个新的页面,并不是SPA
<nuxt-link to="/users"></nuxt-link>
this.$router.push('/users')
1.2.3 动态路由
- 在 Nuxt.js 里面定义带参数的动态路由,需要创建对应的以下划线作为前缀的 Vue 文件 或 目录。
- 获取动态参数
{{$route.params.id}}
1.2.4 路由参数校验
Nuxt.js 可以让你在动态路由对应的页面组件中配置一个validate
方法用于校验动态路由参数的有效性。该函数有一个布尔类型的返回值,如果返回true则表示校验通过,如果返回false则表示校验未通过。
validate(data) { cosole.log(data) return true }
|
1.2.5 嵌套路由
- 添加一个Vue文件,作为父组件
- 添加一个与父组件同名的文件夹来存放子视图组件
- 在父文件中,添加
<nuxt-child></nuxt-child>
组件,用于展示匹配到的子视图
1.3 layouts & pages & components
1.3.1 创建layout
- 去layouts文件夹下面新建一个新的layout组件,例如teachers.vue,并在这个组件中添加
<nuxt />
组件,这样,所有和teachers相关的页面都会有公共的layout
- 给需要用到teachers.vue的组件添加layout属性,并指定需要使用的layout,例如:layout: ‘teachers’
1.3.2 创建特殊layout : error
layouts文件夹下面新建error.vue,error是关键字
1.3.3 新建一个组件
在components文件夹下面新建一个Header.vue组件
引入组件,注意路径的~符号,表示根目录
layout中也能使用组件
1.3.4 样式配置
nuxt.config.js中设置设置全局样式文件路径
css: [ '~/assets/styles/index.css', ],
|
1.4 ElementUI使用
下载npm i element-ui -S
在plugins文件夹下面,创建ElementUI.js文件
import Vue from 'vue' import ElementUI from 'element-ui' Vue.use(ElementUI)
|
在nuxt.config.js中添加配置
css: [ 'element-ui/lib/theme-chalk/index.css' ], plugins: [ {src: '~/plugins/ElementUI', ssr: true } ], build: { vendor: ['element-ui'] }
|
1.5 异步数据
Nuxt.js 扩展了 Vue.js,增加了一个叫 asyncData
的方法,使得我们可以在设置组件的数据之前能异步获取或处理数据。asyncData
方法会在组件(限于页面组件)每次加载之前被调用。它可以在服务端或路由更新之前被调用。所以需要注意这个函数中不能使用this
注意:常规写法如果在created钩子中写异步,是在客户端渲染的而不是在服务端
使用方法:asyncData(context, callback) {callback(null, data)}
context.route.params.xxx
获取参数
callback(new Error(), data)
渲染出错的页面
注意:这个方法在服务器端执行和在客户端执行的区别
1.6 axios的使用
安装npm install --save axios
使用
import axios from 'axios'
asyncData(context, callback) { axios.get('http://localhost:3301/in_theaters') .then(res => { console.log(res); callback(null, {list: res.data}) }) }
|
为防止重复打包,在nuxt.config.js中配置
module.exports = { build: { vendor: ['axios'] } }
|
1.7 小案例(豆瓣电影列表)
json-server
npm install -g json-server
接口i
获取电影列表:http://localhost:3301/in_theaters
(in_theaters可以替换为coming_soon及top250)
获取电影详情:http://localhost:3301/in_theaters/1?_embed=details
1.7.1 豆瓣电影首页创建
movie组件
<template> <div class="movie-header"> <ul> <li> <nuxt-link to="/movie/in_theaters">正在热映</nuxt-link> </li> <li> <nuxt-link to="/movie/coming_soon">即将上映</nuxt-link> </li> <li> <nuxt-link to="/movie/top250">top250</nuxt-link> </li> </ul> </div> </template> <style scoped> .movie-header { position: fixed; top: 0; left: 0; right: 0; } ul { display: flex; justify-content: space-around; align-items: center; padding: 15px 0; background-color: #1e2736; margin: 0; } li { list-style: none; line-height: 30px; height: 30px; } li a { color: white; } li a:hover { color: red; } </style>
|
movie布局
<template> <div class="movie-layout"> <movie-header/> <nuxt/> </div> </template> <script> import MovieHeader from '~/components/MovieHeader' export default { components: { MovieHeader } } </script> <style scoped> .movie-layout { padding-top: 60px; } </style>
|
<template> <div class="movie"> <h1>欢迎访问本网站!</h1> </div> </template> <script> export default { layout: 'movie', head() { return { title: '豆瓣电影', meta: [ {name: 'keywords', content: '电影、经典电影、热映、电视剧、美剧、影评、电影院、电影票、排行、推荐'} ] } } } </script>
|
1.7.2 豆瓣电影列表页面创建
<template> <div class="movie-type"> <nuxt-link v-for="item in movieList" :key="item.id" :to="`/movie/${$route.params.type}/${item.id}`" class="movie-box"> <img :src="item.img" :alt="item.title"> <h4>电影名:{{ item.title }}</h4> <p>上映评分:{{ item.rating === 0 ? '暂无评分' : item.rating }}</p> </nuxt-link> </div> </template> <script> import axios from '~/plugins/axios' export default { layout: 'movie', asyncData(context, callback) { axios.get(`/${context.route.params.type}`) .then( res => { console.log(res); callback(null, {movieList: res.data}) }) } } </script> <style scoped> .movie-type { display: flex; flex-direction: column; align-items: center; } .movie-box { display: flex; flex-direction: column; align-items: center; margin: 20px 0; padding: 10px 0; width: 40%; box-shadow: 0 0 10px #bbb; } .movie-box:hover { box-shadow: rgba(0,0,0,0.3) 0px 19px 60px; } </style>
|
1.7.3 电影详情页面
<template> <div class="detail"> <div class="detail-box"> <img :src="detail.img" :alt="detail.title"> <h3>{{ detail.title }}</h3> <p>电影类型:{{ detail.genres.join(',') }}</p> <p>上映时间:{{ detail.details[0].year }}</p> <p>上映时间:{{ detail.details[0].summary }}</p> </div> </div> </template>
<script> import axios from '~/plugins/axios' export default { layout:'movie', asyncData(context, callback){ axios.get(`/${context.route.params.type}/${context.route.params.id}?_embed=details`).then(res => { console.log(res.data); callback(null,{detail:res.data}); }) } } </script>
<style scoped> .detail { width: 40%; margin: 0 auto; padding: 20px; box-sizing: border-box; box-shadow: 0 0 10px #bbb; } .detail-box { text-align: center; } </style>
|
1.8 SEO优化
1.8.1 全局
在nuxt.config.js配置文件中修改
head: { title: pkg.name, meta: [ { charset: 'utf-8' }, { name: 'viewport', content: 'width=device-width, initial-scale=1' }, { hid: 'description', name: 'description', content: pkg.description } ], link: [ { rel: 'icon', type: 'image/x-icon', href: '/favicon.ico' } ] },
|
1.8.2 局部
head(){ return{ title:'豆瓣电影', meta:[{ 'name':'keywords', 'content': '电影、经典电影、热映、电视剧、美剧、影评、电影院、电影票、排行、推荐' }] } }
|