# Vue 优雅使用技巧(三)-进度条/首屏优化/动画/懒加载

在我们平时使用vue开发手机端时,我们需要三大部分作为前端需要考虑的问题,我们第一部分要考虑移动端布局适配以及字体大小适配之前总结过一次移动端适配问题,第二部分咋样写法上优化自己的组件,指令,混入,尽量使用vue提供的特性去封装自己业务逻辑vue优雅使用技巧(一),第三部分,手机端交互行为体验,是让用户有直观感受,接下来主要总结第三部分

移动端体验比pc端更主要

1. 添加页面切换时,有一个加载进度条,通过vue-router 导航守卫,来控制跳转前和跳转后加载进度条的样式,代码如下

import NProgress from 'nprogress'
import 'nprogress/nprogress.css'
NProgress.configure({ showSpinner: false });


router.beforeEach((to, from, next) => {
    NProgress.start()
    next()
  })
  
router.afterEach(() => {
NProgress.done()
})

注意

  1. 导航守卫可以处理 页面的权限问题(判断是否登录过),一般针对后台管理系统。

        import router from './router'
        import store from './store'
        import { Message } from 'element-ui'
        import NProgress from 'nprogress' // progress bar
        import 'nprogress/nprogress.css' // progress bar style
        import { getToken } from '@/utils/auth' // get token from cookie
        import getPageTitle from '@/utils/get-page-title'
    
        NProgress.configure({ showSpinner: false }) // NProgress Configuration
    
        const whiteList = ['/login', "/regist", "/forget"] // no redirect whitelist
    
        router.beforeEach(async(to, from, next) => {
            // start progress bar
            NProgress.start()
                // set page title
            document.title = getPageTitle(to.meta.title)
                // determine whether the user has logged in
            const hasName = store.getters.name;
            if (hasName) {
                next()
                NProgress.done()
            } else {
                if (whiteList.indexOf(to.path) !== -1) {
                    next()
                } else {
                    Message.error("由于您没有登录,请到登录页登录")
                    setTimeout(() => {
                        next(`/login?redirect=${to.path}`)
                        NProgress.done()
                    }, 1000)
                }
            }
        })
    
        router.afterEach(() => {
            // finish progress bar
            NProgress.done()
        })
    
  2. 手机端一把判断(是否登陆过),通过指令的封装,大部分按钮触发的。
    敬请期待

  3. 还有是直接在 请求接口中拦截(axios),适用 手机端项目和PC端后台系统。

    Vue 中 Axios 的封装和 API 接口的管理

2. 骨架屏(可以自己写/也可以用UI组件库)

什么是骨架屏

什么是骨架屏呢?骨架屏(Skeleton Screen)是指在页面数据加载完成前,先给用户展示出页面的大致结构(灰色占位图),在拿到接口数据后渲染出实际页面内容然后替换掉。Skeleton Screen 是近两年开始流行的加载控件,本质上是界面加载过程中的过渡效果。

假如能在加载前把网页的大概轮廓预先显示,接着再逐渐加载真正内容,这样既降低了用户的焦灼情绪,又能使界面加载过程变得自然通畅,不会造成网页长时间白屏或者闪烁。这就是 Skeleton Screen !

Skeleton Screen 能给人一种页面内容“已经渲染出一部分”的感觉,相较于传统的 loading 效果,在一定程度上可提升用户体验。

我们通过vant 自带骨架屏的 组件举例。

  • 引入第三方的组件,尽量按需导入,需要什么引入什么组件。这样优化代码的打包后的体积
import Vue from "vue";
import "vant/lib/index.css"
import { Tabbar, TabbarItem,Search,Tab, Tabs,Image as VanImage,Tag,Col, Row,List,Cell,PullRefresh,Skeleton,NavBar,Popup,Button } from "vant"
let components = [
    Tabbar,TabbarItem,Search,Tab, Tabs,VanImage,Tag,Col, Row,List,Cell,PullRefresh,Skeleton,NavBar,Popup,Button
]
components.map(component => {
    component.install = Vue => {
        Vue.component(component.name, component);
    }
})
components.map(component => {
    Vue.use(component);
})
  • 组件使用
<van-skeleton title :row="4" :loading="loading">
    <div>实际内容</div>
    <p>
        发射点发射点
        法撒旦发射点发生
        发射点发射点发射点发射点
        啊发射点发射点发撒打发微软微软为
    </p>
</van-skeleton>

具体组件的参数到官网查看 vant骨架屏

  • 效果
骨架屏.jpg

注意:

  • 虽然解决用用户进入当前页面的等待时间的体验感,但是改骨架屏的样式和真实的dom结构没有关联,其实还是不够人性化体验。
  • 骨架屏不是什么场景都要使用,我们不能滥用骨架屏,大部分使用场景,首页模块展示,详情模块展示,因为列表模块展示,因为可以通过展位图(懒加载图片解决)

我们可以自己写骨架屏样式

优势: 因为当前的页面的dom结构也是我们写的。
html

<!-- 真实的渲染结果 -->
 <div class="card">
    <div class="image">
        <img src="https://assets.codepen.io/2002878/wake-up-and-code.jpg" alt="">
    </div>
    <div class="content">
        <h4>CodingStartup</h4>
        <div class="description">
        Lorem ipsum dolor, sit amet consectetur adipisicing elit. Ex quasi enim facere commodi omnis...
        </div>
    </div>
</div>
<!-- 渲染之前的dom -->
  <div class="card loading">
    <div class="image">
 
    </div>
    <div class="content">
        <h4></h4>
        <div class="description">
       
        </div>
    </div>
</div>

css

.card {
  width: 320px;
  background-color: #fff;
  border-radius: 6px;
  overflow: hidden;
  box-shadow: 0px 4px 6px rgba(0, 0, 0, .12);
}

.image {
  height: 200px;
}

.image img {
  display: block;
  width: 100%;
  height: inherit;
  object-fit: cover;
}

.content {
  padding: 2rem 1.8rem;
}

h4 {
  margin: 0 0 1rem;
  font-size: 1.5rem;
  line-height: 1.5rem;
}

.description {
  font-size: 1rem;
  line-height: 1.4rem;
}

/* 定义骨架屏的样式 */
.loading .image,
.loading h4,
.loading .description {
    background: linear-gradient(100deg,
     rgba(255,255,255, 0) 40%,
     rgba(255, 255, 255, 0.5) 50%,
     rgba(255,255,255, 0) 60% 
     ) #ededed;
     background-size: 200% 200%;
     background-position-x: 180%;
     animation:loading 1s ease-in-out infinite;
}

/* 添加动画 */
@keyframes loading {
    0% {
        background-position-x: 180%; 
    }
    100% {
        background-position-x: -20%;
    }
}
/* 文本信息和图片动画时间延迟 */
.loading h4 {
    min-height: 1.6rem;
    border-radius: 4px;
    animation-delay: .1s;
}
.loading .description {
    min-height: 4rem;
    border-radius: 4px;
    animation-delay: .1s;
}

效果:


g2.png

g1.jpg

生成骨架屏方案

  1. vue-skeleton-webpack-plugin插件 https://blog.csdn.net/qq_41725450/article/details/104925718
  • 编写骨架屏的组件
    在这个页面里面我们可以根据需要来编写代码,最好使用样式或者base64的图片,以减少初始的请求。

<!-- detailSkeleton.vue -->
<template>
  <div class="container">
    详情骨架屏
    <img
      src=""
    />
  </div>
</template>
 
<script>
export default {
  components: {},
  data () {
    return {}
  },
  mounted () {},
  methods: {}
}
</script>
 
<style scoped lang="scss">
</style>
  • 编写Skeleton.js文件,与main.js同一级
import Vue from "vue";
import listSkeleton from "./skeleton/listSkeleton.vue";
import detailSkeleton from "./skeleton/detailSkeleton.vue";

export default new Vue({
    components: {
        listSkeleton,
        detailSkeleton
    },
    template:`
        <div>
            <listSkeleton id="listSkeleton" style="display:none;" />
            <detailSkeleton id="detailSkeleton" style="display:none;" />
        </div>
    `

})
  • 基于vue-cli@3.0配置文件
const PrerenderSPAPlugin = require('prerender-spa-plugin');
const SkeletonWebpackPlugin = require('vue-skeleton-webpack-plugin')
const Renderer = PrerenderSPAPlugin.PuppeteerRenderer;
const path = require('path');

// vue.config.js
module.exports = {
    configureWebpack: config => {
        // if (process.env.NODE_ENV !== 'production') return;
        return {
            plugins: [
                // new PrerenderSPAPlugin({
                //     staticDir: path.join(__dirname, 'dist'),
                //     routes: ['/', '/Second', '/Detail'],
                //     renderer: new Renderer({
                //         inject: {
                //             foo: "bar"
                //         },
                //         headless: false,
                //         renderAfterDocumentEvent: 'render-event'
                //     })
                // }),
                // 骨架屏
                new SkeletonWebpackPlugin({
                    webpackConfig: {
                        entry: {
                            app: path.join(__dirname, './src/Skeleton.js')
                        }
                    },
                    minimize: true,
                    quiet: true,
                    router: {
                        mode: 'hash',
                        routes: [
                 { path: /^\/home/, skeletonId: 'listSkeleton' },
                            { path: /^\/Detail/, skeletonId: 'detailSkeleton' }

                        ]
                    }
                })
            ]
        }
    },
    css: {
        // 是否使用css分离插件 ExtractTextPlugin
        extract: true,
        // 开启 CSS source maps?
        sourceMap: false,
        // 启用 CSS modules for all css / pre-processor files.
        modules: false
    }
}
  1. 自动生成指定页面的骨架屏 饿了么团队设计的page-skeleton-webpack-plugin

该方案通过一个 webpack 插件 page-skeleton-webpack-plugin 的方式与项目开发无缝集成,属于在自动生成骨架屏方面做的非常强大的了,并且可以启动 UI 界面专门调整骨架屏,但是在面对复杂的页面也会有不尽如人意的地方,而且生成的骨架屏节点是基于页面本身的结构和 CSS,存在嵌套比较深的情况,体积不会太小,并且只支持 history 模式。

 // webpack.conf.js
 const HtmlWebpackPlugin = require('html-webpack-plugin')
 const { SkeletonPlugin } = require('page-skeleton-webpack-plugin')
 const path = require('path')

 plugins: [
  //...
  new HtmlWebpackPlugin({
    // Your HtmlWebpackPlugin config
  }),
  new SkeletonPlugin({
    pathname: path.resolve(__dirname, `${customPath}`), // 用来存储 shell 文件的地址
    staticDir: path.resolve(__dirname, './dist'), // 最好和 `output.path` 相同
    routes: ['/', '/search'], // 将需要生成骨架屏的路由添加到数组中
  })
 ]
  1. 自动生成骨架屏 draw-page-structure 强烈推荐使用插件
  • 全局安装,npm i draw-page-structure – g

  • dps init 生成配置文件 dps.config.js

  • 修改 dps.config.js 进行相关配置

  • dps start 开始生成骨架屏
    生成dps.config.js


const dpsConfig = {
    url: 'http://192.168.1.96:8080/#/home',      // 待生成骨架屏页面的地址,用百度(https://baidu.com)试试也可以
    output: {
        filepath: '',   // 生成骨架屏的存放页面,一般为项目的入口页面
        injectSelector: '#app'  // 生成的骨架屏插入页面的节点
    },
    // header: {
    //  height: 40,
    //  background: '#1b9af4'
    // },
    // background: '#eee',
    // animation: 'opacity 1s linear infinite;',
    // includeElement: function(node, draw) {
        // 定制某个节点画出来的样子,带上return false
        // if(node.id == 'ui-alert') {
            // 跳过该节点及其子节点
            // return false;
        // }
        // if(node.tagName.toLowerCase() === 'img') {
            // 对该图片生成宽100%,高8%,颜色为红色的色块
            // draw({
                // width: 100,
                // height: 8,
                // left: 0,
                // top: 0,
                // zIndex: 99999999,
                // background: 'red'
            // });
            // return false;
        // } 
    // },
    // writePageStructure: function(html) {
        // 自己处理生成的骨架屏
        // fs.writeFileSync(filepath, html);
        // console.log(html)
    // },
    init: function() {
        // 生成骨架屏之前的操作,比如删除干扰节点
    }
}

module.exports = dpsConfig;
    

生成骨架屏的效果


draw.png

3. 过渡动画

尽情期待

4. 懒加载/防抖/节流 常用的实现

尽情期待

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351