Vue頁面骨架屏

在開發webapp的時候老是會受到首屏加載時間過長的影響,主流的解決方法是在載入完成以前顯示loading圖效果,而一些大公司會配置一套服務端渲染的架構來解決這個問題。考慮到ssr所要解決的一系列問題,愈來愈多的APP採用了「骨架屏」的方式去提高用戶體驗。javascript

小米商城:
小米商城css

1、分析Vue頁面的內容加載過程

vue項目中的入口index.html只有簡單的內容:html

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta http-equiv="Content-Type" content="text/html;charset=UTF-8">
    <title>Document</title>
</head>
<body>
    <div id="root">        
    </div>
    <script type="text/javascript" src="bundle.js"></script></body>
</body>
</html>

當js執行完以後,會用vue渲染成的dom將div#root徹底替換掉。
咱們在div#root中加入模擬骨架屏,在Chrome開發者工具調整網速:vue

<div id="root">
    這裏是骨架屏
</div>


由此可知,將骨架屏內容直接插入div#root中便可實現骨架屏。java

2、使用vue-server-renderer來實現骨架屏

咱們須要骨架屏也是一個單獨的.vue文件,所以咱們須要用到vue-server-renderer。對vue服務端渲染有所瞭解的同窗必定知道,這個插件可以將vue項目在node端打包成一個bundle,而後由bundle生成對應的html。
首先是生成項目:node

.
├── build
│   ├── webpack.config.client.js
│   └── webpack.config.server.js
├── src
│   └── views
│        ├── index
│        │   └── index.vue
│        ├── skeleton
│        │   └── skeleton.vue
│        ├── app.vue
│        ├── index.js
│        └── skeleton-entry.js
├── index.html
└── skeleton.js
└── package.json

vue的服務端渲染通常會用vue-server-renderer將整個項目在node端打包成一份bundle,而這裏咱們只要一份有骨架屏的html,因此會有一個單獨的骨架屏入口文件skeleton-entry.js,一個骨架屏打包webpack配置webpack.config.server.js,而skeleton.js做用是將webpack打包出來的bundle寫入到index.html中。webpack

//skeleton-entry.js
import Vue from 'vue'
import Skeleton from './views/skeleton/skeleton.vue'

export default new Vue({
  components: {
    Skeleton
  },
  template: '<skeleton />'
})
//webpack.config.server.js
const path = require('path')
const { VueLoaderPlugin } = require('vue-loader')
const VueSSRServerPlugin = require('vue-server-renderer/server-plugin')

module.exports = {
  mode: process.env.NODE_ENV,
  target: 'node',
  entry: path.join(__dirname, '../src/skeleton-entry.js'),
  output: {
    path: path.join(__dirname, '../server-dist'),
    filename: 'server.bundle.js',
    libraryTarget: 'commonjs2'
  },
  module: {
    rules: [
      {
        test: /\.vue$/,
        loader: 'vue-loader'
      },
      {
        test: /\.css$/,
        use: [
          'vue-style-loader',
          'css-loader'
        ]
      }    
    ]
  },
  externals: Object.keys(require('../package.json').dependencies),
  resolve: {
    alias: {
      'vue$': 'vue/dist/vue.esm.js'
    }
  },
  plugins: [
    new VueLoaderPlugin(),
    new VueSSRServerPlugin({
      filename: 'skeleton.json'
    })
  ]
}

其中骨架屏的webpack配置由於是node端,因此須要target: 'node' libraryTarget: 'commonjs2'。在VueSSRServerPlugin中,指定了其輸出的json文件名。當執行webpack會在/server-dist目錄下生成一個skeleton.json文件,這個文件記載了骨架屏的內容和樣式,會提供給vue-server-renderer使用。git

//skeleton.js
const fs = require('fs')
const path = require('path')

const createBundleRenderer = require('vue-server-renderer').createBundleRenderer

// 讀取`skeleton.json`,以`index.html`爲模板寫入內容
const renderer = createBundleRenderer(path.join(__dirname, './server-dist/skeleton.json'), {
  template: fs.readFileSync(path.join(__dirname, './index.html'), 'utf-8')
})

// 把上一步模板完成的內容寫入(替換)`index.html`
renderer.renderToString({}, (err, html) => {
  fs.writeFileSync('index.html', html, 'utf-8')
})

注意,做爲模板的html文件,須要在被寫入內容的位置添加<!--vue-ssr-outlet-->佔位符,本例子在div#root裏寫入:github

<div id="root">
 <!--vue-ssr-outlet-->
</div>

最後執行node skeleton就能實現vue的骨架屏。
最終的index.html:web

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta http-equiv="Content-Type" content="text/html;charset=UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
    <meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
    <title>Document</title>
<style data-vue-ssr-id="a7049cb4:0">
.skeleton[data-v-61761ff8] {
  position: relative;
  height: 100%;
  overflow: hidden;
  padding: 15px;
  box-sizing: border-box;
  background: #fff;
}
.skeleton-nav[data-v-61761ff8] {
  height: 45px;
  background: #eee;
  margin-bottom: 15px;
}
.skeleton-swiper[data-v-61761ff8] {
  height: 160px;
  background: #eee;
  margin-bottom: 15px;
}
.skeleton-tabs[data-v-61761ff8] {
  list-style: none;
  padding: 0;
  margin: 0 -15px;
  display: flex;
  flex-wrap: wrap;
}
.skeleton-tabs-item[data-v-61761ff8] {
  width: 25%;
  height: 55px;
  box-sizing: border-box;
  text-align: center;
  margin-bottom: 15px;
}
.skeleton-tabs-item span[data-v-61761ff8] {
  display: inline-block;
  width: 55px;
  height: 55px;
  border-radius: 55px;
  background: #eee;
}
.skeleton-banner[data-v-61761ff8] {
  height: 60px;
  background: #eee;
  margin-bottom: 15px;
}
.skeleton-productions[data-v-61761ff8] {
  height: 20px;
  margin-bottom: 15px;
  background: #eee;
}
</style></head>
<body>
    <div id="root">
        <div data-server-rendered="true" class="skeleton page" data-v-61761ff8><div class="skeleton-nav" data-v-61761ff8></div> <div class="skeleton-swiper" data-v-61761ff8></div> <ul class="skeleton-tabs" data-v-61761ff8><li class="skeleton-tabs-item" data-v-61761ff8><span data-v-61761ff8></span></li><li class="skeleton-tabs-item" data-v-61761ff8><span data-v-61761ff8></span></li><li class="skeleton-tabs-item" data-v-61761ff8><span data-v-61761ff8></span></li><li class="skeleton-tabs-item" data-v-61761ff8><span data-v-61761ff8></span></li><li class="skeleton-tabs-item" data-v-61761ff8><span data-v-61761ff8></span></li><li class="skeleton-tabs-item" data-v-61761ff8><span data-v-61761ff8></span></li><li class="skeleton-tabs-item" data-v-61761ff8><span data-v-61761ff8></span></li><li class="skeleton-tabs-item" data-v-61761ff8><span data-v-61761ff8></span></li></ul> <div class="skeleton-banner" data-v-61761ff8></div> <div class="skeleton-productions" data-v-61761ff8></div><div class="skeleton-productions" data-v-61761ff8></div><div class="skeleton-productions" data-v-61761ff8></div><div class="skeleton-productions" data-v-61761ff8></div><div class="skeleton-productions" data-v-61761ff8></div><div class="skeleton-productions" data-v-61761ff8></div></div>
    </div>
</body>
</html>

看下效果:
圖片描述
效果仍是闊以的。

尾聲

文章開頭小米商城手機頁面就是用的這樣的方法,不一樣的是它的骨架屏是一個base64的圖片。

更多關於vue-server-renderer內容請戳vue-ssr

文章相關代碼已經同步到 Github,歡迎查閱~
相關文章
相關標籤/搜索