taro小程序添加骨架屏的实现代码


Posted in Javascript onNovember 15, 2019

最近做了一些小程序方面的性能优化,如分包加载,添加骨架屏等,这次主要说一下骨架屏的相关内容。

关于骨架屏,有三种方法:

1.直接请UI同学帮忙P张图,当做loading图放上去。这种方法简单粗暴,但是需要请人帮忙~
2.根据每个页面,自己写一套相同的代码来覆盖样式。这种方法的工程量,你懂的~
3.能不能写个组件呢?该组件自动获取元素位置大小信息来渲染,数据返回后将其卸载。

下面主要说第三种方法~

主框架采用taro,一套代码兼容多端,但是今天这个代码,需要考虑兼容性~

根据上面的思路,我们首先要找到骨架屏的容器,然后找到需要P成灰色的元素,获取该元素的位置大小信息,最后就是渲染了~

获取元素,taro提供了API,Taro.createSelectorQuery()。通过这个API返回一个SelectorQuery对象实例,然后再通过selectAll()来查找骨架中带有特定类的class名,查找之后通过boundingClientRect()获取元素的位置大小信息,把这些信息存放在数组中。
我这边写了两个类,一个是skeleton-radius,渲染圆形;一个是skeleton-rect,渲染长方形。后续大家可以自行扩展。

文字有点多,大家可能看着有点云里雾里的,下面上代码~

// 百度小程序目前不支持跨自定义组件的后代选择器: >>>
  // 但是H5使用后代选择器(.the-ancestor .the-descendant)的时候,可以自动识别自定义组件内的后代
  // 微信小程序支持跨自定义组件都后代选择器(.the-ancestor >>> .the-descendant),可修改为如`.${this.props.selector} >>> .${this.props.selector}-radius`
  if (process.env.TARO_ENV === 'weapp') {
    Taro.createSelectorQuery().selectAll(`.${this.props.selector} >>> .${this.props.selector}-radius`)
      .boundingClientRect().exec(rect => {
        that.setState({
          radiusList: rect[0]
        });
      });
  }
  else {
    Taro.createSelectorQuery().selectAll(`.${this.props.selector} .${this.props.selector}-radius`)
    .boundingClientRect().exec(rect => {
      that.setState({
        radiusList: rect[0]
      });
    });
  }

大家也看到上面的注释了,如果要在多端运行,可先判断环境,根据环境使用不同的选择器。上面代码是实现一个圆形的灰色区域~大家如果有多个形状的需求的话,可以简单封装一个函数,这里我就不再细说了,具体的可以到Demo详细查看~
细说一下后代选择器的兼容性问题:

  1. 百度小程序目前不支持跨自定义组件的后代选择器: >>>。
  2. 但是H5使用后代选择器(.the-ancestor .the-descendant)的时候,可以自动识别自定义组件内的后代。使用自定义组件时,外层是否有元素包裹,都可识别到自定义组件内部的指定类选择器。
  3. 微信小程序支持跨自定义组件的后代选择器(.the-ancestor >>> .the-descendant),但使用自定义组件时,外层不能嵌套元素,否则无法识别。

接下来就是渲染了,这个比较简单,直接上代码~这里背景色和将要P成条状等的元素的背景色都可以在使用组件时自定义传入,也可以不传,有默认色~

<View className='skeleton-container' style={{background: `${bgColor}`}}>
      {
        radiusList.map(radiusItem => (
          <View className='skeleton-item skeleton-item-radius' style={{width: `${radiusItem.width}PX`, height: `${radiusItem.height}PX`,
            background: `${itemColor}`, top: `${radiusItem.top}PX`, left: `${radiusItem.left}PX`}}
          />
        ))
      }
      {
        rectList.map(rectItem => (
          <View className='skeleton-item' style={{width: `${rectItem.width}PX`, height: `${rectItem.height}PX`,
            background: `${itemColor}`, top: `${rectItem.top}PX`, left: `${rectItem.left}PX`}}
          />
        ))
      }
    </View>

到这里,组件已经完成了,使用的时候可以直接引入组件,然后传入selector就可以了,注意,由于数据是动态获取的,页面开始为空,这里就需要mock一些假数据来填充页面了~要覆盖的元素类名必须和组件中的图形类保持一致~

<View className='container' style={{fontSize: '20PX'}}>
      {
        showSkeleton && <Skeleton
          selector='skeleton'
          bgColor='pink'
          itemColor='skyblue'
        />
      }
      <View className='skeleton'>
        <View className='userInfo'>
          <Image
            src={userInfo.avatarUrl}
            alt='用户头像'
            className='userInfo-avatar skeleton-radius'
          />
          <Text>{userInfo.nickName}</Text>
        </View>
        <View>
          {
            list.map(item => (
              <View className='skeleton-rect' style={{marginBottom: '30PX'}}>{item}</View>
            ))
          }
        </View>
        {/* 自定义组件外层最好没有元素包裹,否则微信小程序无法识别,但是H5可以识别 */}
        <List />
      </View>
    </View>

看到注释了吗?使用自定义组件时一定要注意噢~自定义组件若被元素报错,微信小程序无法识别到自定义组件内的图形类!!!

最后要说一下适合使用骨架屏的场景:页面结构简单,元素的宽高固定~若元素宽高不固定的话,你写的mock假数据可能和实际渲染出来的页面差距较大,例如瀑布流~

好了,这次的一些收获就到这里了~大家如果有更好的方法,可以留言交流~最后,附上完整的代码地址~

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
Mootools 1.2教程 同时进行多个形变动画
Sep 15 Javascript
location对象的属性和方法应用(解析URL)
Apr 12 Javascript
jQuery实现动画效果的实例代码
May 07 Javascript
node.js中的querystring.parse方法使用说明
Dec 10 Javascript
jQuery中:not选择器用法实例
Dec 30 Javascript
jQuery中contents()方法用法实例
Jan 08 Javascript
js关于命名空间的函数实例
Feb 05 Javascript
全面了解js中的script标签
Jul 04 Javascript
关于Sequelize连接查询时inlude中model和association的区别详解
Feb 27 Javascript
koa-router路由参数和前端路由的结合详解
May 19 Javascript
纯 JS 实现放大缩小拖拽功能(完整代码)
Nov 25 Javascript
js判断浏览器的环境(pc端,移动端,还是微信浏览器)
Dec 24 Javascript
详解Angular Karma测试的持续集成实践
Nov 15 #Javascript
Javascript 类型转换、封闭函数及常见内置对象操作示例
Nov 15 #Javascript
JavaScript定时器常见用法实例分析
Nov 15 #Javascript
解决Layui 表格自适应高度的问题
Nov 15 #Javascript
layui前端时间戳转化实例
Nov 15 #Javascript
JavaScript变量基本使用方法实例分析
Nov 15 #Javascript
JavaScript字符串处理常见操作方法小结
Nov 15 #Javascript
You might like
PHP中MD5函数使用实例代码
2008/06/07 PHP
全新的PDO数据库操作类php版(仅适用Mysql)
2012/07/22 PHP
CodeIgniter多语言实现方法详解
2016/01/20 PHP
Laravel中基于Artisan View扩展包创建及删除应用视图文件的方法
2016/10/08 PHP
php实现的生成迷宫与迷宫寻址算法完整实例
2017/11/06 PHP
thinkphp5.0自定义验证规则使用方法
2017/11/16 PHP
PHP crypt()函数的用法讲解
2019/02/15 PHP
jquery 图片截取工具jquery.imagecropper.js
2010/04/09 Javascript
jquery 事件对象属性小结
2010/04/27 Javascript
自己实现ajax封装示例分享
2014/04/01 Javascript
javascript里使用php代码实例
2014/12/13 Javascript
jquery实现页面关键词高亮显示的方法
2015/03/12 Javascript
nodejs实现发出蜂鸣声音(系统报警声)的方法
2017/01/18 NodeJs
JavaScript之DOM_动力节点Java学院整理
2017/07/03 Javascript
vue 实现数字滚动增加效果的实例代码
2018/07/06 Javascript
详解小程序不同页面之间通讯的解决方案
2018/11/23 Javascript
小程序日历控件使用方法详解
2018/12/29 Javascript
微信小程序 wepy框架与iview-weapp的用法详解
2019/04/10 Javascript
小程序简单两栏瀑布流效果的实现
2019/12/18 Javascript
openLayer4实现动态改变标注图标
2020/08/17 Javascript
[01:06:12]VP vs NIP 2019国际邀请赛小组赛 BO2 第一场 8.15
2019/08/17 DOTA
使用Python的Treq on Twisted来进行HTTP压力测试
2015/04/16 Python
关于不懂Chromedriver如何配置环境变量问题解决方法
2019/06/12 Python
python中用logging实现日志滚动和过期日志删除功能
2019/08/20 Python
如何在python中处理配置文件代码实例
2020/09/27 Python
htnl5利用svg页面高斯模糊的方法
2018/07/20 HTML / CSS
含精油的天然有机化妆品:Indemne
2019/08/27 全球购物
德国受欢迎的旅游和休闲网站:lastminute.de
2019/09/23 全球购物
农民入党思想汇报
2014/01/03 职场文书
保护动物倡议书
2014/04/15 职场文书
小学生母亲节演讲稿
2014/05/07 职场文书
班级团队活动方案
2014/08/14 职场文书
MongoDB连接数据库并创建数据等使用方法
2021/11/27 MongoDB
python和Appium的移动端多设备自动化测试框架
2022/04/26 Python
vue使用watch监听属性变化
2022/04/30 Vue.js
css中:last-child不生效的解决方法
2022/08/05 HTML / CSS