js - 预加载+监听图片资源加载制作进度条
这两天遇到一个新需求:一个一镜到底的h5动画。因为功能的特殊性,就要求我们提前监听页面的静态图片是否全部加载完毕。即处理预加载。
总结下来,下次这种需求需要提前注意以下几点:
一、图片而不是背景图
本来,我所用到的图都是用背景图制作的(因为非接口返回的图片都要求用背景图)。
但是监听静态图片时,后来发现所用的方法监听不到背景图,所以改成了图片。
这是一个坑。
二、获取新加载的图片:Img.load()
1、要监听图片我们要先获取到页面中的所有图片:
jq的方法:find()
var MyImg = $(body).find('img');
很简单的解决了这个问题。
MyImg得到的是目标元素中的所有图片的集合。
2、然后我们要遍历所有的图片,好判断是否加载完毕:
依然是jq的方法:each()
MyImg.each(function(){
//在这里实现 分别对每一个图片的图片加载结果 的监听。
})
3、然后说如何监听图片加载:
万年青jq方法:load()
Img.load(function(){
// 回调里,执行加载完毕一个的记录处理
})
还好这次用的jq写的代码,省了不少事。
为了记录图片加载完毕的个数,我在全局设置一个变量:
var sum = 0;
并准备一个方法让sum累加
function sumAdd(){
sum++;
}
然后load方法的回调里,调用sumAdd
MyImg.load(()=>{
// 用于新加载的图片
sumAdd(1);
});
最后页面加载完成后成功得到 sum=7;(本次案例页面7张图)
三、获取缓存的图片:Img.complete
now
图片加载是能控制了,但是为什么我一刷新他又监听不到了?
哦!原来页面加载完毕后刷新,再展示的图片都是缓存的图,而load又监听不到缓存的图。
要了我的老命了。
于是我又找,什么方法能监听缓存的图啊?
目标锁定了js里的img.complete。注意划重点是js的属性。
所以这里使用上要注意,因为我获取的dom对象是jq的,要转成js的再调complete属性,于是代码直接是:
if(MyImg[0].complete){
// 用于缓存图片
sumAdd();
}
有了之前准备的sumAdd函数做接应,缓存图片的个数也能计算出来了。
至此,一个非常简易又简陋的监听就完成了。
题外话
之前不是这么做的,而是load方法里累加load的个数,complete里累加缓存的个数。
在电脑上测试,两者河井互不犯,相安无事。
但是到手机上发现,会有6张缓存图1张加载图,导致下边要说的加载进度计算错误,先是变成70%,又变回20%。
后来才改成这两种情况都累加到一处了。
四、预加载进度计算并展示
好了,现在需求升级。
为了友好的用户体验,你在后台加载图片的时候,用户不能只看到一个加载中,等半天不知道到底有没有反应。
所以我们要给用户一个及时的反馈,就要获取图片加载的进度。
思路就是 :
加载进度 = 已加载图片资源个数/总的图片资源个数*100+'%';
有了公式,又有之前我们准备的sum(当前加载个数),这个加载进度轻而易举就能得到了:
let progress = Math.ceil(sum/7*100);
7是当前页面中图片的总个数。而Math.ceil向上取整是为了在除不尽的时候不会出现小数点或99.9999%的情况。
当然为了控制万一超过100的情况,只需要保险设置一下:
progress>=100?100:progress
如果加载进度想做成进度条效果,只需要把得到的progress值赋给进度条的宽度即可。
至于进度条怎么做,看我这篇博文。css案例 - 评分效果的星星✨外衣
五、数字动画效果:animate()
后来我又想,进度条旁边加数字展示岂不是更好?
而作为一个有些许追求的程序员,我又不想直接做成数字是多少就生硬切换成多少的效果。
我想做逐渐变化的数字动画效果
这就要另一个方法上场了。
对,没错,还是jq的方法 - animate()方法:
利用其step属性达到动画逐帧改变的效果,也就是文字从1累加渐变成100,而不是生硬的跳转为100。
这个方法的用法是这样的:
$('#loadingTxts').animate({count: progress},{
duration: 350,
step: function() {
if(isNaN(this.count)){
this.count = 0;
return;
}
let boxText = Math.ceil(Number(this.count));
if(boxText >= 100){
boxText = 100;
//接下来执行预加载完毕,页面开始展示
...
}
$('#loadingTxts').text(boxText+'%');// 文字展示
$('.progress').css('width',boxText+'%'); // 进度条宽度设置
}
});
一个坑接一个坑,最后我们被折磨得没了生气的力气。
大致参考代码:
$(function () {
let nameLink = 'https://www.test.com/test/dist/images/',
nameArr = ['1', '2', '3', '4', '5', '6', '7'],
myImgs = $('.baby-box').find('img'),
downAndCache = 0; function progressAni(i){
downAndCache += i;
let progreVal = Math.ceil(downAndCache/nameArr.length*100);
// 文字-单值变化动画
$('#txtsBox').animate({
count: progreVal
},{
duration: 350,
step: function() {
if(isNaN(this.count)){
this.count = 0;
return;
}
let numberTxt = Math.ceil(Number(this.count));
if(numberTxt >= 100){
numberTxt = 100;
$('.loading').fadeOut();
}
$('#txtsBox').text(numberTxt+'%');
$('#progressBox').css('width',numberTxt+'%');
}
});
}
myImgs.each(function(a){
let Img = $(this);
if(Img[0].complete){
// 用于缓存图片
progressAni(1);
}
Img.load(()=>{
// 用于新加载的图片
progressAni(1);
});
Img.error(function() {
// 图片加载失败,可以选择替换图片
console.log(Img.attr('src'));
Img.attr('src',nameLink + nameArr[a] + '.jpg');
});
});
});
2018-09-13 17:32:28
相同思路的 一段其他的处理方法: 2018-10-26 14:00:21
//加载图片
var loadImgIndex = 0,
path = 'images/';
exports.loadImg = function(arr,callback){
var len = arr.length,
callback = callback || function(){};
if(!len){
return;
}
(function loading(){
var img = new Image();
img.onload = function(){
if(loadImgIndex == len-1){
callback();
}else{
loadImgIndex++;
console.log('图片已经加载:' + Math.floor(loadImgIndex/len*100) + '%');
loading();
}
}
img.src = path + arr[loadImgIndex];
img.onerror = function(){
console.log('加载失败');
setTimeout(callback,1000);
}
})(); ---------------------
//作者:kongjunchao159
//来源:CSDN
//原文:https://blog.csdn.net/kongjunchao159/article/details/49587573
//版权声明:本文为博主原创文章,转载请附上博文链接!
还有这个文章,写的讲解也很好。通俗易懂:
https://blog.csdn.net/ssisse/article/details/51655644?utm_source=blogxgwz0
js - 预加载+监听图片资源加载制作进度条的更多相关文章
- jQuery学习(监听DOM加载)
jQuery的extend方法 function njQuery() { } /* njQuery.extend = function (obj) { // 此时此刻的this就是njQuery这个类 ...
- C# NanUI WinFormium监听页面加载开始\结束
个人博客 地址:https://www.wenhaofan.com/article/20190501213608 因为NanUI文档中仅介绍了Formium窗口的监听,但是没有WinFormium相关 ...
- gulp压缩html,css,js文件流程、监听任务、使用gulp创建服务器、同时运行多个任务、反向代理
一.初始化 首先先做一个项目初始化,用来记录你项目中用到的工具 再你项目文件下打开一个控制台,输入命令 yarn init -y 进行初始化 输入命令yarn add gulp -g --- 全局安 ...
- 移动端用js与jquery实时监听输入框值的改动
背景: 在一次移动端H5开发中,需要监听输入框值的实时变动. onchange事件肯定抛弃,因为只能失去焦点才触发. 而keyPress在Android可以触发,iOS不可以. 又不想用Android ...
- js与jquery实时监听输入框值变化方法
本文实例讲述了js与jquery实时监听输入框值的oninput与onpropertychange方法.分享给大家供大家参考.具体如下: 最近做过一个项目,需求是下拉框里自动匹配关键字,具体细节是实时 ...
- js与jquery实时监听输入框值的oninput与onpropertychange方法
文实例讲述了js与jquery实时监听输入框值的oninput与onpropertychange方法.分享给大家供大家参考.具体如下: 最近做过一个项目,需求是下拉框里自动匹配关键字,具体细节是实时监 ...
- iOS-WKWebview 带有进度条加载的ViewController【KVO监听Webview加载进度】
前言 为什么要说 WKWebview,在之前做电子书笔记时已经提过 WKWebview 在iOS8之后已完全替代 Webview,原因就不多说了,主要还是内存过大: 封装 封装一个基于 UIViewC ...
- 监听spring加载完成后事件
有这个想法是在很早以前了,那时的我没有接触什么缓存技术,只知道hibernate有个二级缓存.没有用过memcache,也没有使用过redis. 只懂得将数据放到数组里或者集合里,一直不去销毁它(只有 ...
- 赵雅智_service电话监听2加接通电话录音
步骤: 创建CallStateService继承Service 取得电话服务 监听电话动作 电话监听的对象 没有电话时 停止刻录 重设 刻录完毕一定要释放资源 电话响铃时 从麦克风採集声音 内容输出格 ...
随机推荐
- Java非递归的方式获取目录中所有文件(包括目录)
零.思路解析 对于给出的文件查看其下面的所有目录,将这个目录下的所有目录放入待遍历的目录集合中,每次取出该集合中的目录遍历,如果是目录再次放入该目录中进行遍历. 一.代码 /** * 非递归的方式获取 ...
- PHP usort 使用用户自定义的比较函数对数组中的值进行排序
From: http://www.php100.com/cover/php/2395.html usort (PHP 4, PHP 5) usort — 使用用户自定义的比较函数对数组中的值进行排序 ...
- CSS3和jQuery实现的自定义美化Checkbox和Radiobox
现在经常可以在网络上看到一些非常奇特的表单元素,例如Checkbox复选框和Radiobox单选框,浏览器默认的样式确实是太丑了,而且更让人蛋疼的是各个浏览器的样式还不统一,考虑到现在越来越多的用户使 ...
- Sql Server Snapshot和mysql MVCC
mysql 在一个事务A中插入一条数据 在B事务中查询到的还是以前的数据,可以select *from table,不被锁住 Sql Server 默认级别 读已提交 所以A事务在 X表插入数据, ...
- CentOS7 防火墙配置(关闭)
CentOS7 的防火墙配置跟曾经版本号有非常大差别,经过大量尝试,最终找到解决这个问题的关键 CentOS7这个版本号的防火墙默认使用的是firewall.与之前的版本号使用iptables不一样. ...
- java 坑
时间戳 unix的时间戳和java中的是不同的.unix为10位,java的13位.需要进行转换. long timestamps = 1509783992L; long javaTimstamps ...
- [iOS]XCODE5升级之路
1.Code Sign error: 解决方案:重新下载并安装Provisioning profile 2.错误:Undefined symbols for architecture armv ...
- Objective-C语法之NSSortDescriptor
main.m #import <Foundation/Foundation.h> #import "Person.h" /** NSSortDescriptor 可以实 ...
- thinkphp 在阿里云上的nginx.config配置
# For more information on configuration, see: # * Official English Documentation: http://nginx.org/e ...
- 如何使用 MasterPage
MasterPageFile母版页 刚开始学,什么都不懂,看到了这段代码,才促使自己去研究MasterPageFile到底是什么含义.<%@ Page Language="C#&quo ...