对开发更友好的前端骨架屏自动生成方案

本文介绍了马蜂窝电商前端团队实现的一种自动化生成骨架屏的方法,旨在提升用户体验,减少白屏时间。通过Puppeteer操控Chrome生成骨架屏,解决手写骨架屏的维护难题。文章对比了现有的几种骨架屏方案,提出了一种读取配置文件,通过Puppeteer获取DOM结构并生成骨架屏的方案,该方案已应用于多页面项目中。
摘要由CSDN通过智能技术生成

点击上方“马蜂窝技术”,关注订阅更多优质内容

一份来自 Akamai 的研究报告显示,在对 1048 名网购户进行采访后发现:

  • 约 47% 的用户期望他们的页面在两秒之内加载完成。

  • 如果页面加载时间超过 3s,约 40% 的用户会选择离开或关闭页面。


一直以来,为了提升用户在页面加载时的体验,无论是 Web 还是 iOS、Android 的应用中,前端开发工程师都做了许多工作。除了解决如何让网页展现速度更快的问题,还有很重要的一点就是提升用户对加载等待时间的感知。「菊花图」以及由其衍生出的各种加载动画就是一类常见的解决方案,相信无论是开发者还是用户对下面这个图标都不会陌生:

本文将要介绍的「骨架屏」则被视为菊花图升级版的方案。受现有骨架屏方案的启发,马蜂窝电商前端研发团队实现了一种自动化生成骨架屏的方法,并在马蜂窝商城的多个页面中实现应用,取得了不错的效果。

一、

什么是骨架屏

骨架屏可以理解为在页面数据尚未返回或页面未完成完全渲染前,先给用户呈现一个由灰白块组成的当前页面大致结构,让用户产生页面正在逐渐渲染的感受,从而使加载过程从视觉上变得流畅。生成后的骨架屏页面如下图所示:

骨架屏的主要优势为:

1.用户避免看到长时间的白页
2.可以获知页面的大体结构,减小用户认为页面出错而离开的机率
3.与菊花图相比视觉更加流畅

二、

常见的前端骨架屏方案

在选择骨架屏之前,我们也考虑了一些其他的方法,比如能否通过服务端渲染(SSR)的方式来避开前端白屏时间的问题。但发现需要涉及项目过多,还会涉及服务的构建与部署;或是通过 prerender-spa-plugin 提供简单的预呈现,它对 SPA 支持友好,但需要额外的 webpack 配置,且因为包源的问题,下载时间过长,有时还会莫名失败,等等,都因为种种原因最终放弃。

经过一系列调研后,我们对业界常见的几种骨架屏解决方案,以及它们的优势、不足进行了一个简单的梳理。

1. UI 骨架屏图

即通过 UI 提供符合页面首页样式的图来充当骨架屏,将骨架屏 base64 图片插入 root 根节点,在 webpack 打包时嵌入项目中。

这是一种简单粗暴的方法,实现起来比较容易。但缺点也很明显,就是需要 UI 设计师支持和开发介入,不能自动生成。

2.  手写骨架屏

即通过手写 HTML、CSS 的方式为目标页定制骨架屏。这种方式可以做到对页面真实样式的复刻。不过一旦由于各种原因导致页面样式发生改变,就需要再改一遍骨架屏的样式和布局,极大增加了维护的成本。

3. 自动生成静态骨架屏 

目前比较受关注的是饿了么开源的插件 page-skeleton-webpack-plugin,其具体实现原理为:

  • 生成骨架屏

通过 Puppeteer 操控 handless Chrome 打开需要生成的骨架屏页面,在等待页面加载完成之后,保留页面布局样式的前提下,通过对页面中元素进行增删,对已有元素通过层叠样式进行覆盖,使其展示为灰白块。然后将修改后的 HTML 和 CSS 提取出来,将页面分为不同的块区域,例如文本块、图片块、按钮块、SVG、伪类元素块等,分别对每个块进行处理,使其尽量与原页面保持一致。这里用到了 Puppeteer  page 实例的 addScriptTag 方法来将处理块的脚本插入到 headless Chrome 打开的页面当中。

实际生成的骨架屏页面与原页面可能还会存在差距,插件通过 memory-fs 将骨架屏写入内存中,可以通过预览页面对生成的骨架屏进行二次编辑和效果预览,修改完成后点击生成按钮就能生成一份新的骨架屏写入到项目中。

借一张图来说明:

  • 插入骨架屏

骨架屏的 DOM 结构和 CSS 通过离线生成后,在构建时注入模板 (EJS) 中的节点下面,插入到 HTML 是在 after-emit 钩子函数中进行。

page-skeleton-webpack-plguin 生成骨架屏的方案可以根据项目中不同的路由页面生成相应的骨架屏页面,并将骨架屏页面通过 webpack 打包到对应的静态路由页面中。

它的不足之处在于:

  • 实际使用过程中无法监听接口返回导致生成骨架屏的时机是否准确

  • 生成的页面与业务人员写的结构质量有直接关系,经常出现需要手工二次调整的情况

在这样的背景下,马蜂窝电商研发前端团队

UNIAPP是一个基于Vue.js构建跨平台应用框架,支持原生体验、性能优化以及丰富的组件生态,能够同时生成iOS和Android应用。对于自动生成骨架(skeleton loading),特别是适用于多页面的应用,UNIAPP提供了良好的支持。 ### 骨架的作用 骨架是一种在数据加载过程中显示的临时界面,它模拟了最终内容的形式和布局,帮助用户理解数据正在加载,并减少等待感。这对于提高用户体验特别重要,在网络延迟较长的情况下尤其如此。 ### 实现多页自动生成骨架的方法 #### 使用 UNIAPP 的 `<uni-load-more>` 组件 UNIAPP 提供了一个 `<uni-load-more>` 组件,用于实现列表或数据加载多功能。通过这个组件,你可以很容易地添加加载状态的展示,比如在列表顶部预加载部分数据,并展示骨架效果。当数据真正加载完毕时,骨架会被实际的数据替换掉。 示例如下: ```html <view class="content"> <uni-load-more :status="loadStatus" @click="onLoadMoreClick"> <!-- 数据列表渲染 --> <template v-for="(item, index) in items"> <!-- 每个 item 的展示逻辑 --> </template> <!-- 当前状态下的骨架展示 --> <div v-if="!items.length && loadStatus !== 'noMore'" class="skeleton-loading"> <!-- 根据需求调整 skeleton 的样式 --> <div class="skeleton-item"></div> <div class="skeleton-item"></div> <!-- 多 skeleton 元素... --> </div> </uni-load-more> </view> ``` 在这个示例中,`items` 数组包含了当前页面的数据列表,`loadStatus` 属性控制着加载状态的显示,包括 `loading`、`noMore` 和默认的 `normal` 状态。当数据尚未加载或已经加载完所有数据时,会显示相应的骨架。 #### 利用 Vue.js 或者 UNIAPP 自带的状态管理库(如 Vuex) 为了在不同页面间共享加载状态,可以使用 Vue.js 的状态管理工具如 Vuex,或者直接利用 UNIAPP 提供的状态管理解决方案。这有助于统一管理和新各个页面的加载状态信息,进而动态控制骨架的显示和隐藏。 #### 页面间切换与状态保存 为了实现在切换页面时保持加载状态和显示适当的骨架,可以在需要的地方监听路由变化,并新全局或特定页面的加载状态。这通常涉及在 `beforeRouteEnter`, `beforeRouteUpdate` 或 `beforeRouteLeave` 中进行状态处理。 ### 示例代码段 假设我们有多个使用上述骨架策略的页面: ```javascript // store.js (Vuex 示例) import { createStore } from 'vuex'; export default new createStore({ state() { return { // 添加一个全局的加载状态属性 globalLoading: false, }; }, }); ``` ```vue <!-- 页面 A.vue --> <template> <uni-load-more :status="loadStatus" @click="onLoadMoreClick"> <!-- ...页面A的UI和数据展示... --> <div v-if="globalLoading && loadStatus === 'loading'" class="skeleton-loading"> <!-- ...页面A的骨架展示... --> </div> </uni-load-more> </template> <script setup> import { useStore } from 'vuex'; const store = useStore(); // ...其他代码... </script> ``` ### 相关问题: 1. **如何优化骨架的加载速度**? 可以考虑提前加载部分数据并逐步填充,避免一次性加载大量数据导致的性能瓶颈。 2. **如何在不同设备上适配骨架的尺寸**? 使用响应式设计,确保骨架元素的比例和大小适合各种幕尺寸和分辨率。 3. **如何提高用户体验,使得用户在等待数据加载时不感到烦躁**? 设计简洁直观的骨架界面,增加交互性(如滑动反馈等),并提供进度提示或动画,让用户知道加载正在进行中。 通过以上步骤和策略,你可以在 UNIAPP 开发的多页应用中实现高效、美观且增强用户体验的自动生成骨架功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值