正文
通用(也称同构)的JavaScript已经成为JavaScript社区很常用的一个术语。通用的JavaScript用来形容可以在客户端执行,也可在服务端执行的Javascript代码。
作者的其他文章
很多现代的JavaScript框架,比如
Vue.js
, 旨在构建单页应用(SPA)。单页应用的优势在于,改善用户体验,让网页速度更快,像APP一样流畅,即使更新。虽然单页应用优点很多,但是由于依赖多导致首屏渲染慢,无法做seo优化也是让人头疼的问题。
服务端渲染是指,提前将页面在服务器端渲染好,当浏览器请求服务器时,直接返回渲染好的html页面返回。
构建服务端渲染的JavaScript程序多少有些无趣,在开始编码之前,需要大量的基础配置。因此,解决vue.js服务端渲染问题的
Nuxt.js
产生了。
Nuxt.js 概要
简而言之,Nuxt.js是帮助Vue.js轻松完成服务端渲染工作的框架。Nuxt.js预设了服务端渲染所需要的各种配置,如异步数据,中间件,路由。它好比是
Angular Universal
之于
Angular
,
Next.js
之于
React
。
如
Nuxt.js文档
所说,通过对客户端/服务端基础架构的抽象,Nuxt.js 让开发者专注于页面的UI渲染。
静态文件生成器
Nuxt.js的一个重要功能是,通过
generate
命令,生成静态站点。类似于流行的静态生成工具
Jekyll
。
Nuxt.js 内部依赖
除了Vue.js 2.0之外,Nuxt.js集成了如下模块:
Vue-Router
,
Vue-Meta
和
Vuex
(仅在使用
Vuex 状态树配置项
时引入)。 这样的好处在于,不需要手工配置依赖,不需要同时在客户端和服务端配置相同的库。 Nuxt.js在包含如上依赖的情况下,总大小仍然保持在
28kb min+gzip
(如果使用了 Vuex 特性的话为 31kb)。
另外,Nuxt.js 使用
Webpack
和
vue-loader
、
babel-loader
来处理代码的自动化构建工作(如打包、代码分层、压缩等等)。
工作原理
当你访问一个基于Nuxt.js构建的页面时,发生了的事情如下:
-
当用户访问应用程序, 如果store中定义了
nuxtServerInit
action
,Nuxt.js将调用它更新store。
-
接下来,将加载即将访问页面所依赖的任何
中间件
。Nuxt首先从
nuxt.config.js
这个文件中,加载全局依赖的中间件,之后检测每个相应页面对应的
布局文件
,最后,检测布局文件下子组件依赖的中间件。以上是中间件的加载顺序。
-
如果要访问的路由是一个动态路由, 且有一个相应的
validate()
方法
路由的validate 方法
,讲进行路由校验。
-
之后, Nuxt.js 调用
asyncData()
和
fetch()
方法,在渲染页面之前加载异步数据。
asyncData()
方法用于异步获取数据,并将fetch回来的数据,在服务端渲染到页面。 用
fetch()
方法取回的将数据在渲染页面之前填入store。
-
最后一步, 将所有数据渲染到页面。
下图阐述了 Nuxt.js 应用一个完整的服务器请求到渲染的流程,摘自官网:
使用 Nuxt.js 创建一个静态网站
下面让我们动手创建一个基于Nuxt.js简单的静态博客。我们的发送的请求,返回 mock 的JSON数据。
完成下面例子,你需要了解基础的 vue.js 知识。如果你是个新手,你可以通过Jack Franklin的
getting started guide
了解 Vue.js 2.0。同时,我将使用ES6语法,你可以参考
www.sitepoint.com/tag/es6/
重温ES6语法。
我们的 Demo 最终效果如下:
本文中代码可参照
GitHub
, demo 地址
如下
。
基础配置
开始使用 Nuxt.js 最简单的方式是使用 Nuxt.js 团队自己开发的脚手架。我们可以使用
vue-cli
快速创建我们的项目 (
ssr-blog
):
`vue init nuxt/starter ssr-blog`
提示:
如果你没有安装过vue-cli,请先通过
npm install -g vue-cli
命令安装vue-cli。
之后,我们将安装项目的依赖:
cd ssr-blog
npm install
现在我们启动程序:
`npm run dev`
如果正确启动, 你能访问
http://localhost:3000
,展示的页面是 Nuxt.js 模板的初始页面。你也可以通过查看页面源代码,验证页面所展示的一切内容,都是服务端渲染好的。
下面,我们简单配置下
nuxt.config.js
,包含以下选项:
// ./nuxt.config.js
module.exports = {
/*
* Headers of the page
*/
head: {
titleTemplate: '%s | Awesome JS SSR Blog',
// ...
link: [
// ...
{
rel: 'stylesheet',
href: 'https://cdnjs.cloudflare.com/ajax/libs/bulma/0.4.2/css/bulma.min.css'
}
]
},
// ...
}
在如上配置文件下,我们使用
titleTemplate
字段
title
变量指定文章题目,在渲染之前用title变量值替换掉
%s
这个占位,填充到
titleTemplate
。
同时,我也使用了 CSS 框架,
Bulma
, 预设一些样式。通过
link
配置项。
提示:
Nuxt.js使用
vue-meta
更新我们的 html headers 信息。所以,我们可以看看 meta 具体的配置项,优化页面 html 信息。
现在,我们可以通过几个步骤,完成博客的页面和功能。
使用 Layouts
首先,我们将为我们所有的页面定义一个通用的基本布局。我们通过修改
layouts/default.vue
文件,更新 main Nuxt.js layout:
<!-- ./layouts/default.vue -->
<template>
<div>
<!-- navigation -->
<nav class="nav has-shadow">
<div class="container">
<div class="nav-left">
<nuxt-link to="/" class="nav-item">
Awesome JS SSR Blog!
</nuxt-link>
<nuxt-link active-class="is-active" to="/" class="nav-item is-tab" exact>Home</nuxt-link>
<nuxt-link active-class="is-active" to="/about" class="nav-item is-tab" exact>About</nuxt-link>
</div>
</div>
</nav>
<!-- /navigation -->
<!-- displays the page component -->
<nuxt/>
</div>
</template>
在我们通用的布局里,我们仅仅对页面添加导航栏,我们通过
component
进一步完成具体页面模块的定制。你可以查看
components-nuxt-link
进一步了解。
在创建布局时
component
非常重要,它决定具体页面展示的元素。
当然,component也可以做更多事情,比如定义通用组件和错误页面,但是我们的博客很简单,不需要这些功能。强烈建议阅读
Nuxt.js documentation on views
,你可以通过这篇文章了解更多 Nuxt.js 特性。
简单的页面和路由
Nuxt.js 页面是以
单文件组件
形式组织目录结构。 Nuxt.js 自动找到目录下每个
.vue
文件,并添加到页面中。
创建博客主页
我们可以通过修改
index.vue
文件修改主页, 通过 Nuxt.js 创建的文件如下:
<!-- ./pages/index.vue -->
<template>
<div>
<section class="hero is-medium is-primary is-bold">
<div class="hero-body">
<div class="container">
<h1 class="title">
Welcome to the JavaScript SSR Blog.
</h1>
<h2 class="subtitle">
Hope you find something you like.
</h2>
</div>
</div>
</section>
</div>
</template>
<script>
export default {
head: {
title: 'Home'
}
}
</script>
如前所述,在渲染之前,题目将自动填充至文件。
我们现在可以刷新页面,看看主页的变化。
创建 About 页面
Nuxt.js 还有一个优秀的特性,监听文件夹下文件的更改,所以,在文件更改时,不需要重启应用更新。
来,我们添加一个简单的
about.vue
页面:
<!-- ./pages/about.vue -->
<template>
<div class="main-content">
<div class="container">
<h2 class="title is-2">About this website.</h2>
<p>Curabitur accumsan turpis pharetra <strong>augue tincidunt</strong> blandit. Quisque condimentum maximus mi, sit amet commodo arcu rutrum id. Proin pretium urna vel cursus venenatis. Suspendisse potenti. Etiam mattis sem rhoncus lacus dapibus facilisis. Donec at dignissim dui. Ut et neque nisl.</p>
<br>
<h4 class="title is-4">What we hope to achieve:</h4>
<ul>
<li>In fermentum leo eu lectus mollis, quis dictum mi aliquet.</li>
<li>Morbi eu nulla lobortis, lobortis est in, fringilla felis.</li>
<li>Aliquam nec felis in sapien venenatis viverra fermentum nec lectus.</li>
<li>Ut non enim metus.</li>
</ul>
</div>
</div>
</template>
<script>
export default {
head: {
title: 'About'
}
}
</script>
现在我们访问
http://localhost:3000/about
看看about页面,无需重启,非常方便。
在主页展示文章列表
我们的首页在没有内容的时候展示如上, 所以下一步,我们要在
index.vue
上添加博客列表这个组件。
首先,我们需要把 JSON 格式的文章保存在服务根目录下。文件可以从
这里
下载,或者你可以复制下面的 JSON 到根目录文件夹
posts.json
下:
[
{
"id": 4,
"title": "Building universal JS apps with Nuxt.js",
"summary": "Get introduced to Nuxt.js, and build great SSR Apps with Vue.js.",
"content": "<p>Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.</p>",
"author": "Jane Doe",
"published": "08:00 - 07/06/2017"
},
{
"id": 3,
"title": "Great SSR Use cases",
"summary": "See simple and rich server rendered JavaScript apps.",
"content": "<p>Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.</p>",
"author": "Jane Doe",
"published": "17:00 - 06/06/2017"
},
{
"id": 2,
"title": "SSR in Vue.js",
"summary": "Learn about SSR in Vue.js, and where Nuxt.js can make it all faster.",
"content": "<p>Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.</p>",
"author": "Jane Doe",
"published": "13:00 - 06/06/2017"
},
{
"id": 1,
"title": "Introduction to SSR",
"summary": "Learn about SSR in JavaScript and how it can be super cool.",
"content": "<p>Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.</p>",
"author": "John Doe",
"published": "11:00 - 06/06/2017"
}
]
提示:
理想情况下,我们应该从通过 API 获取文章数据。例如,
Contentful
是就是一个提供cms后台服务的网站。
components 存放在
components
文件夹下,我们可以创建如下组件:
<!-- ./components/Posts.vue -->
<template>
<section class="main-content">
<div class="container">
<h1 class="title has-text-centered">
Recent Posts.
</h1>
<div class="columns is-multiline">
<div class="column is-half" v-for="post in posts">
<div class="card">
<header class="card-header">
<p class="card-header-title">
{{ post.title }}
</p>
</header>
<div class="card-content">
<div class="content">
{{ post.summary }}
<br>
<small>
by <strong>{{ post.author}}</strong>
\\ {{ post.published }}
</small>
</div>
</div>
<footer class="card-footer">
<nuxt-link :to="`/post/${post.id}`"
class="card-footer-item">
Read More
</nuxt-link>
</footer>
</div>
</div>
</div>
</div>
</section>
</template>
<script>
import posts from '~/posts.json'
export default {
name: 'posts',
data () {
return { posts }
}
}
</script>
我们引入 JSON 文件充当异步数据,通过
v-for
指令循环列表,取出我们需要的属性填充进组件模板展示。