Vue2.x项目骨架屏注入(下)

1,127 阅读3分钟

骨架屏注入实践二:dps(draw-page-structure)

前提背景:

vue-skeleton-webpack-plugin通过预渲染手动书写的代码生成相应的骨架屏,通过 vueSSR 结合 webpack 在构建时渲染写好的 vue 骨架屏组件,将预渲染生成的 DOM 节点和相关样式插入到最终输出的 html 中,然后预渲染生成骨架屏所需的 DOM 节点,但由于该方案不够灵活、可控;因此针对Vue2.x项目骨架屏注入(上)的方案进行优化——网页骨架屏自动生成(dps)。

dps 具体操作步骤

1、安装dps

npm i draw-page-structure -D

2、在具体某个页面中需要生成骨架屏的页面添加:

mounted() {
    // 注意:生成骨架屏后,把骨架屏代码贴到vue文件后,下面代码需要注释掉,否则每次进来又生成html插入页面
    setTimeout(() => {
        createSkeletonHTML({
        // 此段代码实际无效
            background: 'red',
            animation: 'opacity 1s linear infinite;'
        }).then((skeletonHTML) => {
        // 此处输出来的就是骨架屏节点,可以自定义骨架屏节点内容
            console.log(skeletonHTML);
        }).catch((e) => {
            console.error(e);
        });
    }, 5000);
},

3、npm run dev 在浏览器中运行后,可在控制台输出当前页面骨架屏节点,复制添加到应用页面,如:

src/commercialActivity/index.vue

<template lang="html">
    <div class="commercialActivity">
    <!--为了交互好看就用了transition-->
        <transition name="skeleton">
            <div v-if="showSkeleton">
            <!--骨架屏节点-->
                <div class="_ __" style="height:100%;z-index:990;background:#fff"></div><div class="_" style="height:2.099%;top:3.748%;left:6.667%;width:73.667%"></div><div class="_" style="height:2.099%;top:3.748%;left:81.067%;width:12.267%"></div><div class="_" style="height:1.799%;top:8.096%;left:6.667%;width:34.492%"></div><div class="_" style="height:1.799%;top:8.096%;left:74.133%;width:19.200%"></div><div class="_" style="height:4.498%;top:14.768%;left:9.333%;width:7.467%"></div><div class="_" style="height:2.249%;top:15.967%;left:26.975%;width:40.000%"></div><div class="_" style="height:2.249%;top:15.892%;left:69.333%;width:4.000%;border-radius:50%"></div><div class="_" style="height:2.099%;top:15.967%;left:76.000%;width:16.000%"></div><div class="_" style="height:2.249%;top:27.361%;left:1.867%;width:3.950%"></div><div class="_" style="height:10.195%;top:23.388%;left:8.483%;width:17.908%"></div><div class="_" style="height:2.249%;top:23.763%;left:29.058%;width:66.938%"></div><div class="_" style="height:2.399%;top:26.912%;left:30.125%;width:6.400%;border-radius:2px"></div><div class="_" style="height:1.799%;top:27.211%;left:74.246%;width:21.750%"></div><div class="_" style="height:2.249%;top:30.960%;left:29.058%;width:66.938%"></div><div class="_" style="height:2.399%;top:37.031%;left:40.858%;width:12.800%"></div><div class="_" style="height:1.150%;top:37.830%;left:55.942%;width:3.200%"></div><div class="_" style="height:2.849%;top:42.429%;left:35.067%;width:29.867%"></div><div class="_" style="height:2.249%;top:95.610%;left:1.867%;width:4.000%"></div><div class="_" style="height:6.447%;top:93.403%;left:6.975%;width:6.400%"></div><div class="_" style="height:6.447%;top:93.403%;left:73.333%;width:26.667%"></div>
            </div>
        </transition>
....
    </div>
</template>
<style>
._{
position:fixed
z-index:999;
<!--由于createSkeletonHTML方法中,background: 'red',里边不生效,粘贴代码时手动修改一下样式,否则是undefined,看不到颜色及效果-->
background:#efefef; 
}
.__{
top:0%;
left:0%;
width:100%;
}
</style>

app.styl

// 骨架过渡
.skeleton-enter-active
  transition opacity .6s
.skeleton-enter
  opacity 0
.skeleton-leave-active
  transition opacity .6s
.skeleton-leave-to
  opacity 0
  

骨架是在页面没渲染完成,就是接口没返回之前展示,接口返回骨架隐藏。因此需要把mounted里边生成骨架屏的代码进行注释,否则每次页面运行完毕都会自动生成骨架屏节点,自动插入页面,导致多份骨架屏重叠的现象,也因此对骨架屏节点进行v-if="showSkeleton"控制:

src/commercialActivity/commercialActivity.js

data() {
        return {
            showSkeleton: true, // 是否显示骨架
        }
    }
methods: {
        // 滚动下拉下拉加载
        infinite(val) {
            return _.debounce(() => {
            // 请求接口的地方
                this.getProductInfo({
                    activity_id: this.activityId,
                    page: this.listQery.page,
                    ser: val ? val.search : ''
                }).then(() => {
                // 接口返回骨架隐藏
                    this.showSkeleton = false;
                    if (!this.errorType.type) {
                        this.$refs.infiniteLoading.stateChanger.loaded();
                    } else {
                        this.$refs.infiniteLoading.stateChanger.complete();
                    }
                    if (this.hasMore) {
                        this.distance = -Infinity;
                    } else {
                        this.$refs.infiniteLoading.stateChanger.complete();
                    }
                }).catch((err) => {
                // 接口返回骨架隐藏
                    this.showSkeleton = false;
                    console.log(err);
                });
                if (this.isAllChecked) {
                    this.isAllChecked = true;
                }
            }, 500);
        },
}

4、dps是把当前可视的页面生成骨架 dps设计的规则如下:

    1. 只遍历可见区域可见的 DOM 节点,包括:非隐藏元素、宽高大于 0 的元素、非透明元素、内容不是空格的元素、位于浏览窗口可见区域内的元素等;
    1. 针对(背景)图片、文字、表单项、音频视频、Canvas、自定义特征的块等区域来生成颜色块;
    1. 页面节点使用的样式不可控,所以不可取 style 的尺寸相关的值,可通过 getBoundingClientRect 获取节点宽、高、距离视口距离的绝对值,计算出与当前设备的宽高对应的百分比作为颜色块的单位,来适配不同设备

5、由于dps生成的页面骨架是运行放在页面节点中,而不是index.html中的首屏骨架,所以看到的交互是看到'白屏->页面骨架->正常页面',实际dps只是单纯的生成页面的骨架的的代码。所以在首屏index.html页面加入loading的交互(具体看项目是否需要首屏loading的交互):

<body>
    <div id="app">
        <!-- 样式写在该地方是为了能够当页面过来了这个app下的所有内容都给替换掉 -->
        <style>
            #app {
              height: 100%;
              margin: 0px;
              padding: 0px;
            }
            #loader-wrapper {
              position: fixed;
              top: 0;
              left: 0;
              width: 100%;
              height: 100%;
            }
            #loader {
              display: block;
              position: relative;
              left: 50%;
              top: 50%;
              width: 150px;
              height: 150px;
              margin: -75px 0 0 -75px;
              border-radius: 50%;
              border: 3px solid transparent;
              border-top-color: red;
              -webkit-animation: spin 2s linear infinite;
              -ms-animation: spin 2s linear infinite;
              -moz-animation: spin 2s linear infinite;
              -o-animation: spin 2s linear infinite;
              animation: spin 2s linear infinite;
            }
            #loader:before {
              content: "";
              position: absolute;
              top: 5px;
              left: 5px;
              right: 5px;
              bottom: 5px;
              border-radius: 50%;
              border: 3px solid transparent;
              border-top-color: red;
              -webkit-animation: spin 3s linear infinite;
              -moz-animation: spin 3s linear infinite;
              -o-animation: spin 3s linear infinite;
              -ms-animation: spin 3s linear infinite;
              animation: spin 3s linear infinite;
            }
            #loader:after {
              content: "";
              position: absolute;
              top: 15px;
              left: 15px;
              right: 15px;
              bottom: 15px;
              border-radius: 50%;
              border: 3px solid transparent;
              border-top-color: red;
              -moz-animation: spin 1.5s linear infinite;
              -o-animation: spin 1.5s linear infinite;
              -ms-animation: spin 1.5s linear infinite;
              -webkit-animation: spin 1.5s linear infinite;
              animation: spin 1.5s linear infinite;
            }
            #loader-wrapper .fengche-logo {
              display: block;
              position: absolute;
              left: 50%;
              top: 50%;
              width: 60px;
              height: 60px;
              margin: -30px 0 0 -30px;
            }
            #loader-wrapper .fengche-logo img {
              width: 100%;
            }
            @-webkit-keyframes spin {
              0% {
                -webkit-transform: rotate(0deg);
                -ms-transform: rotate(0deg);
                transform: rotate(0deg);
              }
              100% {
                -webkit-transform: rotate(360deg);
                -ms-transform: rotate(360deg);
                transform: rotate(360deg);
              }
            }
            @keyframes spin {
              0% {
                -webkit-transform: rotate(0deg);
                -ms-transform: rotate(0deg);
                transform: rotate(0deg);
              }
              100% {
                -webkit-transform: rotate(360deg);
                -ms-transform: rotate(360deg);
                transform: rotate(360deg);
              }
            }
</style>
        <div id="loader-wrapper">
        <div id="loader"></div>
        <div class="fengche-logo">
            <img src="" />
        </div>
        </div>
    </div>
    <!-- built files will be auto injected -->
</body>

运行后,页面的交互为:首屏loading->页面骨架屏->数据请求回来后的页面正常ui交互

6、当项目中使用的手淘lib-flexible或者px进行了rem的处理和使用一些ui框架,例如vant,那么使用dps的话,直接生成的%骨架屏,就不适用了,就需要自定义骨架屏,把%处理为rem,这样页面的骨架屏ui交互才正常

mounted() {
    setTimeout(() => {
    /* 此举为自定义骨架屏,处理%转为rem,项目的根字体大小=37.5px */
            createSkeletonHTML({}).then(skeletonHTML => {
                // 最好自定义高度比例较大的,兼容各种较长的屏幕,例如375 * 1000 (iphoneX - 375 * 812)
                const skeletonWidth = document.body.clientWidth // 屏幕宽度
                const skeletonHeight = document.body.clientHeight // 屏幕高度
                const skeletonTop = 0 // 骨架屏从哪截取(width:375px下top的px值,不包括header的44px)
                let cacheHtml = JSON.parse(JSON.stringify(skeletonHTML))
                cacheHtml = cacheHtml.replace(/<style>\S+<\/style>/, '') // 去掉style,提取到公共
                // 下面代码是将百分比转换为rem:height(px) = val * clientHeight/100
                cacheHtml = cacheHtml.replace(/height:\d*\.*\d*%/g, (val) => {
                    return `height:${(skeletonHeight * val.slice(7, -1) / 3750).toFixed(6)}rem`
                })
                cacheHtml = cacheHtml.replace(/top:\d*\.*\d*%/g, (val) => {
                    return `top:${(skeletonHeight * val.slice(4, -1) / 3750).toFixed(6)}rem`
                })
                cacheHtml = cacheHtml.replace(/left:\d*\.*\d*%/g, (val) => {
                    return `left:${(skeletonWidth * val.slice(5, -1) / 3750).toFixed(6)}rem`
                })
                cacheHtml = cacheHtml.replace(/width:\d*\.*\d*%/g, (val) => {
                    return `width:${(skeletonWidth * val.slice(6, -1) / 3750).toFixed(6)}rem`
                })
                console.log('首屏骨架:', cacheHtml)
                // 骨架屏截取,去掉header44px(屏幕宽度375px下)+自定义的skeletonTop
                const skeletonCutTop = (skeletonTop + 44) / 37.5
                cacheHtml = cacheHtml.replace(/<div class="_" style="\S+<\/div>/g, (val) => {
                    let isCut = true
                    val = val.replace(/top:\d*\.*\d*rem/g, (topRem) => {
                        if (topRem.slice(4, -3) < skeletonCutTop) {
                            isCut = false
                            return topRem
                        }
                        return `top:${(topRem.slice(4, -3) - (44 / 37.5)).toFixed(6)}rem`
                    })
                    return isCut ? val : ''
                })
                // 骨架屏背景截取+下移
                cacheHtml = cacheHtml.replace(/<div class="_ __" style="\S+<\/div>/g, (val) => {
                    return `<div class="_ __" style="height:${((skeletonHeight - skeletonTop) / 37.5).toFixed(6)}rem;top:${(skeletonTop / 37.5).toFixed(6)}rem;z-index:990;background:#fff"></div>`
                })
                // 微信环境下header隐藏的处理
                cacheHtml = `<div class="skeleton-box"><div :class="[$store.getters.isWx||$store.getters.isApp?'':'not-wx','skeleton-content']">${cacheHtml}</div></div>`
                console.log('页面骨架:', cacheHtml)
            }).catch(e => {
                console.error(e)
            })
        }, 8000)
}

注意事项1:

lib-flexible方案限制font-size最大54px;更好的兼容是改为跟项目的max-width:1280px一样;max-width:1280px应是固定不应转rem,px改为大写PX

function refreshRem(){
    var width = docEl.getBoundingClientRect().width;
    if (width / dpr > 1280) {
        width = 1280 * dpr;
    }
    var rem = width / 10;
    docEl.style.fontSize = rem + 'px';
    flexible.rem = win.rem = rem;
}

注意事项2:

顶部类似搜索框的vant组件可不用骨架遮挡

const skeletonTop = 48 // 骨架屏从哪截取(width:375px下不需截取的height,不包括header的44px)

注意事项3:

vant组件也需要对应转rem,postcss.config.js改为:

module.exports = ({ file }) => {
    let remUnit
    if (file && file.dirname && file.dirname.indexOf('vant') > -1) {
        remUnit = 37.5
    } else {
        remUnit = 75
    }
    return {
        plugins: {
            'postcss-px2rem-exclude': {
                remUnit,
            }
        }
    }
}

注意事项4:

一般静态页面是不加骨架屏的,因为没有接口请求,本身加载速度很快,没有必要骨架屏,而且骨架屏也是个静态,只是为了有请求的交互体验提高!