一、什么是骨架屏鹰祸?
骨架屏可以理解為是在需要等待加載內(nèi)容的位置提供一個占位圖形組合甫窟,
描繪了當前頁面的大致框架的骨架屏頁面,然后骨架屏中各個占位部分被實際資源完全替換蛙婴,
這個過程中用戶會覺得內(nèi)容正在逐漸加載即將呈現(xiàn)粗井,降低了用戶的焦躁情緒,使得加載過程主觀上變得流暢街图。
二浇衬、何時使用
1、網(wǎng)絡(luò)較慢餐济,需要長時間等待加載處理的情況下焕窝。
2煤惩、圖文信息內(nèi)容較多的列表/卡片中。
三、對比菊花圖
第一個為骨架屏用押,第二個為菊花圖款违,第三個為無優(yōu)化油猫,可以看到相比于傳統(tǒng)的菊花圖會在感官上覺得內(nèi)容出現(xiàn)的流暢而不突兀夕凝,體驗更加優(yōu)良。
四鸽照、生成骨架屏的方法
1螺捐、手寫HTML、CSS的方式為目標頁定制骨架屏 做法可以參考<Vue頁面骨架屏注入實踐>矮燎,主要思路就是使用 vue-server-renderer 這個本來用于服務(wù)端渲染的插件定血,用來把我們寫的.vue
文件處理為HTML
,插入到頁面模板的掛載點中诞外,完成骨架屏的注入澜沟。這種方式不甚文明,如果頁面樣式改變了峡谊,還得改一遍骨架屏倔喂,增加了維護成本。 骨架屏的樣式實現(xiàn)參考 CodePen
2靖苇、 使用圖片作為骨架屏; 簡單暴力班缰,讓UI同學花點功夫吧哈哈贤壁;小米商城的移動端頁面采用的就是這個方法,它是使用了一個Base64的圖片來作為骨架屏埠忘。
3脾拆、 自動生成并自動插入靜態(tài)骨架屏 這種方法跟第一種方法類似馒索,不過是自動生成骨架屏,可以關(guān)注下餓了么開源的插件 page-skeleton-webpack-plugin 名船,它根據(jù)項目中不同的路由頁面生成相應(yīng)的骨架屏頁面绰上,并將骨架屏頁面通過 webpack 打包到對應(yīng)的靜態(tài)路由頁面中,不過要注意的是這個插件目前只支持history方式的路由渠驼,不支持hash方式蜈块,且目前只支持首頁的骨架屏,并沒有組件級的局部骨架屏實現(xiàn)迷扇,作者說以后會有計劃實現(xiàn)(issue9)百揭。
4、另外還有個插件 vue-skeleton-webpack-plugin蜓席,它將插入骨架屏的方式由手動改為自動
器一,原理在構(gòu)建時使用 Vue 預(yù)渲染功能,將骨架屏組件的渲染結(jié)果 HTML 片段插入 HTML 頁面模版的掛載點中厨内,將樣式內(nèi)聯(lián)到 head
標簽中祈秕。這個插件可以給單頁面的不同路由設(shè)置不同的骨架屏,也可以給多頁面設(shè)置雏胃,同時為了開發(fā)時調(diào)試方便请毛,會將骨架屏作為路由寫入router中,可謂是相當體貼了丑掺。
4.1获印、vue-server-renderer
4.1.1 分析Vue頁面的內(nèi)容加載過程
為了簡單起見,我們使用vue-cli
搭配webpack-simple
這個模板來新建項目:
vue init webpack-simple vue-skeleton
這時我們便獲得了一個最基本的Vue項目:
.
├── package.json
├── src
│ ├── App.vue
│ ├── assets
│ └── main.js
├── index.html
└── webpack.conf.js
安裝完了依賴以后街州,便可以通過npm run dev
去運行這個項目了兼丰。但是,在運行項目之前唆缴,我們先看看入口的html文件里面都寫了些什么鳍征。
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<title>vue-skeleton</title>
</head>
<body>
<div id="app"></div>
<script src="/dist/build.js"></script>
</body>
</html>
可以看到,DOM里面有且僅有一個div#app
面徽,當js被執(zhí)行完成之后艳丛,此div#app
會被整個替換掉,因此趟紊,我們可以來做一下實驗氮双,在此div里面添加一些內(nèi)容:
<div id="app">
<p>Hello skeleton</p>
<p>Hello skeleton</p>
<p>Hello skeleton</p>
</div>
打開chrome的開發(fā)者工具,在Network
里面找到throttle
功能霎匈,調(diào)節(jié)網(wǎng)速為“Slow 3G”戴差,刷新頁面,就能看到頁面先是展示了三句“Hello skeleton”铛嘱,待js加載完了才會替換為原本要展示的內(nèi)容暖释。
現(xiàn)在袭厂,我們對于如何在Vue頁面實現(xiàn)骨架屏,已經(jīng)有了一個很清晰的思路——在div#app
內(nèi)直接插入骨架屏相關(guān)內(nèi)容即可球匕。
4.1.2 易維護的方案
顯然纹磺,手動在div#app
里面寫入骨架屏內(nèi)容是不科學的,我們需要一個擴展性強且自動化的易維護方案亮曹。既然是在Vue項目里橄杨,我們當然希望所謂的骨架屏也是一個.vue
文件,它能夠在構(gòu)建時由工具自動注入到div#app
里面乾忱。
首先讥珍,我們在/src
目錄下新建一個Skeleton.vue
文件,其內(nèi)容如下:
<template>
<div class="skeleton page">
<div class="skeleton-nav"></div>
<div class="skeleton-swiper"></div>
<ul class="skeleton-tabs">
<li v-for="i in 8" class="skeleton-tabs-item"><span></span></li>
</ul>
<div class="skeleton-banner"></div>
<div v-for="i in 6" class="skeleton-productions"></div>
</div>
</template>
<style>
.skeleton {
position: relative;
height: 100%;
overflow: hidden;
padding: 15px;
box-sizing: border-box;
background: #fff;
}
.skeleton-nav {
height: 45px;
background: #eee;
margin-bottom: 15px;
}
.skeleton-swiper {
height: 160px;
background: #eee;
margin-bottom: 15px;
}
.skeleton-tabs {
list-style: none;
padding: 0;
margin: 0 -15px;
display: flex;
flex-wrap: wrap;
}
.skeleton-tabs-item {
width: 25%;
height: 55px;
box-sizing: border-box;
text-align: center;
margin-bottom: 15px;
}
.skeleton-tabs-item span {
display: inline-block;
width: 55px;
height: 55px;
border-radius: 55px;
background: #eee;
}
.skeleton-banner {
height: 60px;
background: #eee;
margin-bottom: 15px;
}
.skeleton-productions {
height: 20px;
margin-bottom: 15px;
background: #eee;
}
</style>
接下來窄瘟,再新建一個skeleton.entry.js
入口文件:
import Vue from 'vue'
import Skeleton from './Skeleton.vue'
export default new Vue({
components: {
Skeleton
},
template: '<skeleton />'
})
在完成了骨架屏的準備之后衷佃,就輪到一個關(guān)鍵插件vue-server-renderer
登場了。該插件本用于服務(wù)端渲染蹄葱,但是在這個例子里氏义,我們主要利用它能夠把.vue
文件處理成html
和css
字符串的功能,來完成骨架屏的注入图云,流程如下:
4.1.3 方案實現(xiàn)
接下來惯悠,在根目錄下新建一個skeleton.js
,該文件即將被用于往index.html
內(nèi)插入骨架屏竣况。
const fs = require('fs')
const { resolve } = require('path')
const createBundleRenderer = require('vue-server-renderer').createBundleRenderer
// 讀取`skeleton.json`克婶,以`index.html`為模板寫入內(nèi)容
const renderer = createBundleRenderer(resolve(__dirname, './dist/skeleton.json'), {
template: fs.readFileSync(resolve(__dirname, './index.html'), 'utf-8')
})
// 把上一步模板完成的內(nèi)容寫入(替換)`index.html`
renderer.renderToString({}, (err, html) => {
fs.writeFileSync('index.html', html, 'utf-8')
})
注意,作為模板的
html
文件丹泉,需要在被寫入內(nèi)容的位置添加``占位符情萤,本例子在div#app
里寫入:<div id="app"> <!--vue-ssr-outlet--> </div>
根據(jù)流程圖,我們還需要在根目錄新建一個webpack.skeleton.conf.js
文件摹恨,以專門用來進行骨架屏的構(gòu)建筋岛。
const path = require('path')
const webpack = require('webpack')
const nodeExternals = require('webpack-node-externals')
const VueSSRServerPlugin = require('vue-server-renderer/server-plugin')
module.exports = {
target: 'node',
entry: {
skeleton: './src/skeleton.entry.js'
},
output: {
path: path.resolve(__dirname, './dist'),
publicPath: '/dist/',
filename: '[name].js',
libraryTarget: 'commonjs2'
},
module: {
rules: [
{
test: /\.css$/,
use: [
'vue-style-loader',
'css-loader'
]
},
{
test: /\.vue$/,
loader: 'vue-loader'
}
]
},
externals: nodeExternals({
allowlist: /\.css$/
}),
resolve: {
alias: {
'vue$': 'vue/dist/vue.esm.js'
},
extensions: ['*', '.js', '.vue', '.json']
},
plugins: [
new VueSSRServerPlugin({
filename: 'skeleton.json'
})
]
}
可以看到,該配置文件和普通的配置文件基本完全一致晒哄,主要的區(qū)別在于其target: 'node'
睁宰,配置了externals
,以及在plugins
里面加入了VueSSRServerPlugin
寝凌。在VueSSRServerPlugin
中柒傻,指定了其輸出的json文件名。
我們可以通過運行下列指令webpack --config ./webpack.skeleton.conf.js
较木,在/dist
目錄下生成一個skeleton.json
文件,
運行node skeleton.js
诅愚,就可以完成骨架屏的注入了
在package.json
中配置ske
"scripts": {
"dev": "cross-env NODE_ENV=development webpack-dev-server --open --hot",
"build": "cross-env NODE_ENV=production webpack --progress --hide-modules",
"ske": "webpack --config ./webpack.skeleton.conf.js && node skeleton.js"
},
運行ske
npm run ske
skeleton.json
這個文件在記載了骨架屏的內(nèi)容和樣式,會提供給vue-server-renderer
使用。
<html lang="en">
<head>
<meta charset="utf-8">
<title>vue-skeleton</title>
<style data-vue-ssr-id="742d88be:0">
.skeleton {
position: relative;
height: 100%;
overflow: hidden;
padding: 15px;
box-sizing: border-box;
background: #fff;
}
.skeleton-nav {
height: 45px;
background: #eee;
margin-bottom: 15px;
}
.skeleton-swiper {
height: 160px;
background: #eee;
margin-bottom: 15px;
}
.skeleton-tabs {
list-style: none;
padding: 0;
margin: 0 -15px;
display: flex;
flex-wrap: wrap;
}
.skeleton-tabs-item {
width: 25%;
height: 55px;
box-sizing: border-box;
text-align: center;
margin-bottom: 15px;
}
.skeleton-tabs-item span {
display: inline-block;
width: 55px;
height: 55px;
border-radius: 55px;
background: #eee;
}
.skeleton-banner {
height: 60px;
background: #eee;
margin-bottom: 15px;
}
.skeleton-productions {
height: 20px;
margin-bottom: 15px;
background: #eee;
}
</style></head>
<body>
<div id="app">
<div data-server-rendered="true" class="skeleton page"><div class="skeleton-nav"></div> <div class="skeleton-swiper"></div> <ul class="skeleton-tabs"><li class="skeleton-tabs-item"><span></span></li><li class="skeleton-tabs-item"><span></span></li><li class="skeleton-tabs-item"><span></span></li><li class="skeleton-tabs-item"><span></span></li><li class="skeleton-tabs-item"><span></span></li><li class="skeleton-tabs-item"><span></span></li><li class="skeleton-tabs-item"><span></span></li><li class="skeleton-tabs-item"><span></span></li></ul> <div class="skeleton-banner"></div> <div class="skeleton-productions"></div><div class="skeleton-productions"></div><div class="skeleton-productions"></div><div class="skeleton-productions"></div><div class="skeleton-productions"></div><div class="skeleton-productions"></div></div>
</div>
<script src="/dist/build.js"></script>
</body>
</html>
可以看到违孝,骨架屏的樣式通過<style></style>
標簽直接被插入,而骨架屏的內(nèi)容也被放置在div#app
之間泳赋。當然雌桑,我們還可以進一步處理,把這些內(nèi)容都壓縮一下祖今。改寫skeleton.js
校坑,在里面添加html-minifier
:
...
+ const htmlMinifier = require('html-minifier')
...
renderer.renderToString({}, (err, html) => {
+ html = htmlMinifier.minify(html, {
+ collapseWhitespace: true,
+ minifyCSS: true
+ })
fs.writeFileSync('index.html', html, 'utf-8')
})
來看看效果:
4.2、page-skeleton-webpack-plugin
4.2.1安裝page-skeleton-webpack-plugin
npm install --save-dev page-skeleton-webpack-plugin
4.2.2 根目錄創(chuàng)建shell
文件夾(文件夾名字可以自己定義千诬,但是要和vue.config.js中儲存shell文件地址一致)耍目,用于儲存shell
文件,也就是page-skeleton-webpack-plugin
自動生成生成的骨架屏html
文件
4.2.3 在index.html入口文件添加<!-- shell -->
占位符
<div id="app">
<!-- shell -->
</div>
若想要更改占位符,修改位置:修改node_modules/page-skeleton-webpack-plugin/src/util/index.js
const outputSkeletonScreen = async (originHtml, options, log) => {
const { pathname, staticDir, routes } = options
return Promise.all(routes.map(async (route) => {
const trimedRoute = route.replace(/\//g, '')
const filePath = path.join(pathname, trimedRoute ? `${trimedRoute}.html` : 'index.html')
const html = await promisify(fs.readFile)(filePath, 'utf-8')
const finalHtml = originHtml.replace('<!-- shell -->', html) # 修改此處徐绑,只要保持和index.html入口文件占位符一致即可
const outputDir = path.join(staticDir, route)
const outputFile = path.join(outputDir, 'index.html')
await fse.ensureDir(outputDir)
await promisify(fs.writeFile)(outputFile, finalHtml, 'utf-8')
log(`write ${outputFile} successfully in ${route}`)
return Promise.resolve()
}))
}
4.2.4 創(chuàng)建vue.config.js
const { SkeletonPlugin } = require('page-skeleton-webpack-plugin')
const path = require('path')
module.exports = {
configureWebpack: {
plugins: [
new SkeletonPlugin({
pathname: path.resolve(__dirname, './shell'), // 用來存儲 shell 文件的地址
staticDir: path.resolve(__dirname, './dist'), // 最好和 `output.path` 相同
routes: ['/','/about'], // 將需要生成骨架屏的路由添加到數(shù)組中
image:{ // 可配置骨架屏元素樣式
color:"#333333",
shape:"circle"
}
})
],
},
chainWebpack: (config) => { // 解決vue-cli3腳手架創(chuàng)建的項目壓縮html 干掉<!-- shell -->導(dǎo)致骨架屏不生效
if (process.env.NODE_ENV !== 'development') {
config.plugin('html').tap(opts => {
opts[0].minify.removeComments = false
return opts
})
}
},
};
4.2.5 運行項目
npm run serve
報錯解決辦法Error:
listen EADDRINUSE: address already in use :::8989
修復(fù)vue-cli3.0項目端口被占用的bug
// 修改node_modules/page-skeleton-webpack-plugin/src/skeletonPlugin.js
if (!this.server) {
const server = this.server = new Server(this.options) // eslint-disable-line no-multi-assign
server.listen().catch(err => server.log.warn(err))
}
4.2.6 生成骨架屏
在瀏覽器打開頁面邪驮,通過 Ctrl|Cmd + enter 呼出插件交互界面,或者在在瀏覽器的 JavaScript 控制臺內(nèi)輸入toggleBar 呼出交互界面
骨架屏生成中傲茄,需要一小會兒時間
骨架屏生成好后毅访,會跳轉(zhuǎn)到以下頁面
保存骨架屏后,會在項目中的shell目錄下生成相關(guān)骨架頁面
4.2.7查看骨架屏效果
npm run build
4.3盘榨、vue-skeleton-webpack-plugin
4.3.1 安裝vue-skeleton-webpack-plugin
npm install vue-skeleton-webpack-plugin
4.3.2 創(chuàng)建模板文件喻粹,如果不同的路由界面顯示不同的模板可以創(chuàng)建過個模板文件,我在src
的common文件夾下面創(chuàng)建了skeleton文件夾并創(chuàng)建三個文件,這樣文件樣式可以根據(jù)自己需求自定義
Skeleton1.vue
<template>
<div class="skeleton-wrapper">
<header class="skeleton-header"></header>
<section class="skeleton-block">
<img src="">
<img src="">
</section>
</div>
</template>
<script>
export default {
name: 'skeleton'
};
</script>
<style scoped>
.skeleton-header {
height: 152px;
background: grey;
margin-top: 60px;
width: 152px;
margin: 60px auto;
}
.skeleton-block {
display: flex;
flex-direction: column;
padding-top: 8px;
}
</style>
Skeleton2.vue
<template>
<div class="skeleton-wrapper">
<header class="skeleton-header"></header>
<section class="skeleton-block">
<img src="">
<img src="">
</section>
</div>
</template>
<script>
export default {
name: 'skeleton'
};
</script>
<style scoped>
.skeleton-header {
height: 152px;
background: grey;
margin-top: 60px;
width: 152px;
margin: 60px auto;
}
.skeleton-block {
display: flex;
flex-direction: column;
padding-top: 8px;
}
</style>
entry-skeleton.js
import Vue from 'vue'
import Skeleton1 from './Skeleton1'
import Skeleton2 from './Skeleton2'
export default new Vue({
components: {
Skeleton1,
Skeleton2
},
template: `
<div>
<skeleton1 id="skeleton1" style="display:none"/>
<skeleton2 id="skeleton2" style="display:none"/>
</div>
`
})
4.3.3創(chuàng)建vue.config.js
const SkeletonWebpackPlugin = require('vue-skeleton-webpack-plugin')
const path = require('path')
module.exports = {
configureWebpack: (config) => {
config.plugins.push(new SkeletonWebpackPlugin({
webpackConfig: {
entry: {
app: path.join(__dirname, './src/skeleton.js')
}
},
minimize: true,
quiet: true,
router: {
mode: 'hash',
routes: [
{ path: '/', skeletonId: 'skeleton1' },
{ path:'/about', skeletonId: 'skeleton2' }
]
}
}))
},
// css相關(guān)配置
css: {
// 是否使用css分離插件 ExtractTextPlugin
extract: true,
// 開啟 CSS source maps?
sourceMap: false,
// 啟用 CSS modules for all css / pre-processor files.
modules: false
},
// 在開發(fā)模式下分離css樣式草巡,讓骨架屏的css在開發(fā)模式下生效
css: {
extract: true
}
}
vue-skeleton-webpack-plugin插件參數(shù)說明
webpackConfig 必填守呜,渲染 skeleton 的 webpack 配置對象
insertAfter 選填,渲染 DOM 結(jié)果插入位置山憨,默認值為字符串 '<div id="app">'
也可以傳入 Function查乒,方法簽名為 insertAfter(entryKey: string): string,返回值為掛載點字符串
quiet 選填萍歉,在服務(wù)端渲染時是否需要輸出信息到控制臺
router 選填 SPA 下配置各個路由路徑對應(yīng)的 Skeleton
mode 選填 路由模式侣颂,兩個有效值 history|hash
routes 選填 路由數(shù)組,其中每個路由對象包含兩個屬性:
path 路由路徑 string|RegExp
skeletonId Skeleton DOM 的 id string
minimize 選填 SPA 下是否需要壓縮注入 HTML 的 JS 代碼
來源:
https://juejin.im/post/5b79a2786fb9a01a18267362
https://segmentfault.com/a/1190000014832185