前情

uni-app是我比较喜欢的跨平台框架,它能开发小程序/H5/APP(安卓/iOS),重要的是对前端开发友好,自带的IDE可视化的运行和打包也让开发体验也非常棒,公司项目就是主推uni-app,为了用户体验对于耗时操作,如接口请求或者异步的API调用都会添加loading效果

原生API使用持性

uni-app项目本身自带有原生交互反馈的API,如showToast、showLoading、showModal、showActionSheet,各有应用场景,而其中showLoading就是用于显示一个加载中效果的,同时跟他配套的还有hideLoading,用于隐藏已经显示的loading效果

showLoading特性:永远只会有一个,如果同时调用多次只会显示最后调用的那一个

hideLoading特性:调用它会关掉页面上的正在显示的loading

思考:

  • 因特性引起的一些使用问题

假设一个场景,我页面上同时调起二个接口,一个接口都会调起一个loading ,最后显示的是后调接口的loading,但是此时前一个接口回来了,会调用hideLoading接口,就会隐藏loading,其实第二个接口还没有回来,导致loading被关了,测试代码如下:

const test0 = () => {
uni.showLoading({
title: '加载中...0',
mask: true
});
setTimeout(() => {
uni.hideLoading();
}, 1000);
} const test1 = () => {
uni.showLoading({
title: '加载中...1',
mask: true
});
setTimeout(() => {
uni.hideLoading();
}, 2000);
}
test0();
test1();
  • 使用体验问题

在我们做接口请求的时候,接口速度是会受用户的网速影响,网速好就接口返回快,网络差就接口返回慢,这里会有一个问题,如果用户网络好那loading会闪一下就没了,用户都没看到是什么东西,在一定层面上给用户带来不好的体验

解决方案

  • 解决因特性引起的使用问题

因hideLoading并不会识别当前显示是由谁唤起的loading,导致无法识别当前要隐藏的是哪一个Loading,那我们就封装下代码,记录已经唤起的loading,在调用loading的时候通过传参指定要隐藏的是哪一个 loading就行了

  • 优化体验问题

我们无法确认用户的网络状态,那我们在唤起loading的时候是否可以做一个延时显示,假设我们延时为300ms,如果发现300ms内有唤起当前loading的hideloading调用,那当前loading也就不用显示了,但是这又有一个问题,如果网络正好回复时间了310,那一样是闪一下,所以我们再加一个loading的最小显示时间

uni-app项目一想到要全局操作,首选第一想到就是事件通信了,此处我们基于uni-app自带的事件通信$on、$emit来实现一个能解决上面缺陷的loading显示与隐藏方案,完整代码如下:

//  用于存储记录当前已有loading
let loadingObj = {};
// loading需要显示的基准时间
let loadingDelayShow = 300;
// loading显示的最小时间
let loadingDelayHide = 1000;
let loadingTimer = {}; // 默认 loading 配置
const loadingOptionsDefault = {
mask: true
} /**
* 初始化loading
* 基于事件通知实现 loading 的显示与隐藏
*/
export const initLoading = (options = '数据加载中...') => {
// 监听显示 loading 事件
uni.$off('showLoading');
// optionsIn是loading的配置,参考showLoading的配置,
// 其中key用于存储当前是什么哪一个loading,用于与hideLoading配合使用
uni.$on('showLoading', (optionsIn = options, key) => {
console.log('---- showLoading ----:', optionsIn, key);
// 如果传入的是字符串,则将其作为 title
const loadingOptions = typeof optionsIn === 'string'
? { ...loadingOptionsDefault, title: optionsIn }
: { ...loadingOptionsDefault, ...optionsIn };
if (!loadingObj[key]) {
loadingObj[key] = {
show: false,
startTime: Date.now()
};
}
// 如果300ms内又调用了hideloading,则无需显示loading
loadingTimer[key] = setTimeout(() => {
if (loadingObj[key] && !loadingObj[key].show) {
loadingObj[key].show = true;
loadingObj[key].startTime = Date.now();
uni.showLoading({...loadingOptions});
}
}, loadingDelayShow);
}); // 监听隐藏 loading 事件
uni.$off('hideLoading');
uni.$on('hideLoading', (key) => {
if (loadingObj[key]) {
loadingObj[key].show = false;
clearTimeout(loadingTimer[key]);
}
if (isCanHide(key)) {
const { startTime } = loadingObj[key];
if (Date.now() - startTime >= loadingDelayHide) {
resetLoading();
} else {
setTimeout(() => {
if (isCanHide(key)) {
resetLoading();
}
}, loadingDelayHide - (Date.now() - startTime));
}
}
}); // 监听重置 loading 事件
uni.$off('resetLoading');
uni.$on('resetLoading', () => {
try {
resetLoading();
} catch (err){
console.log('---- resetLoading ----', err);
}
});
} // 判断要不要隐藏loading
const isCanHide = (key) => {
if (!loadingObj[key].show && Object.keys(loadingObj).length === 1) {
return true;
}
// 或者所有的loading对象的show都是false
for (const key in loadingObj) {
if (loadingObj[key].show) {
return false;
}
}
return true;
} const resetLoading = () => {
loadingObj = {}
uni.hideLoading();
} // 导出事件名称常量
export const LOADING_EVENTS = {
SHOW: 'showLoading',
HIDE: 'hideLoading',
RESET: 'resetLoading'
}; // 导出默认对象,包含事件名称常量
export default {
LOADING_EVENTS
};

使用方式:

在项目根目录main.js中初始化

import { initLoading } from '@/utils/loading';
initLoading()

在需要的地方调用

// 显示loading
uni.$emit('showLoading', '提交订单中...', 'xxx'); // 隐藏loading
uni.$emit('hideLoading', 'xxx');

使用注意事项

  • 其中xxx是key,要确保唯一,实在是怕重复的话可以引入第三方的uuid来做
  • 显示与隐藏要配套使用,不然记录的loading不会清空或者状态不会变,会导致问题

小结

这是我的 uni-app项目的loading使用方案,已使用在生产中,还没有发现什么问题,如果后续发现什么问题再做更新吧,解决方案千千万,世上没有最好,只有更好,如果你有更好的解决方案,可以分享出来,或者你发现此方案有什么问题,也可以提出来一起探讨。

uni-app项目loading显示方案的更多相关文章

  1. 基于React-Native0.55.4的语音识别项目全栈方案

    移动端的API能力验证方案与PC端不一样!不一样!!不一样!!! 即使需要使用的API都存在,也不一定能用,这一点和PC端是有很大区别的,国内的手机系统虽然都是基于Android,但几乎都会经过各大厂 ...

  2. 移动APP项目研发流程及版本规划(转)

    一个移动APP项目研发规模可大可小,但都离不开以下几个成员:产品经理.ui设计师.前端开发.后端开发.测试等.如何合理安排项目成员工作.确保项目顺利进行呢?一个清晰合理的项目研发流程控制很重要. 项目 ...

  3. 论APP测试中黑盒测试方案的重要性?

    运筹帷幄之中,决胜千里之外.古人足不出户,通过正确的部署就能决定千里之外战争的胜利!而于测试人员而言,制定正确的测试方案,就是日后测试就是是否顺利的决定性因素. 在整个测试过程中,对测试人员.资源以及 ...

  4. 第一次,触碰Web App项目,栽过的那些坑。

    此项目是一个IPad上的Web App项目,页面的滚动用了最新的IScroll 5.0 插件, 确实是挺潮的. 项目用时 1个月 完成的, 准备今天晚上上线. 这是年前的最后一篇文章了,与众位博友分享 ...

  5. uni app中使用自定义图标库

    项目中难免会用到自定义图标,那在uni app中应该怎么使用呢? 首先, 将图标目录放在static资源目录下: 在main.js中引入就可以全局使用了 import '@/static/icon-o ...

  6. 用户识别APP项目开发计划书

    用户识别APP项目开发计划书        项目介绍: 用户识别APP,通过在有限时间内引导用户A交互,提取用户的行为特征,然后将APP交给用户X(可能是A也可能是陌生人),在1分钟内引导X交互,判断 ...

  7. csdn的app打开贴子显示空白?

    csdn或者虎扑的app打开贴子显示空白,卸载后重装仍然有同样的问题. 可能是android系统的WebView版本太落后. 打开应用市场,更新WebView就可以解决了.

  8. vue隐藏APP启动时显示的{{}}

    vue隐藏APP启动时显示的{{}} vue组件在编译好之前会显示{{msg}},在官网上找到这个

  9. 到底为什么你的APP项目烂尾了?

    你正在经历迷茫.纠结,或者愤怒.痛苦的情绪,因为,你的APP项目已经或将要烂尾了. 目前的状况只有3种: 项目一直拖到现在,并且很可能继续拖下去 项目在开发期间不断上涨成本 项目完成,BUG多多,不能 ...

  10. 在xcode运行编译时,编译成功,但项目中显示缺少该文件,这是只要关闭重启xcode即可。

    在xcode运行编译时,编译成功,但项目中显示缺少该文件,这是只要关闭重启xcode即可.

随机推荐

  1. Alice与Bob-整数分解问题脚本实现

    题目: 密码学历史中,有两位知名的杰出人物,Alice和Bob.他们的爱情经过置换和轮加密也难以混淆,即使是没有身份认证也可以知根知底.就像在数学王国中的素数一样,孤傲又热情.下面是一个大整数:985 ...

  2. nginx集群同步方案

    之前公司同事写过rsync加触发nginx reload脚本,适合nginx配置内容完全一致的情况. 今天写一个同步指定文件的脚本,修改完主服务器.使用scp传输到其他nginx服务器上重启NGINX ...

  3. 包装类面试题--java进阶day05

    1.面试题 如下两个输出,请问分别是true还是false呢? 答案: 当范围在-128~127时,对象相同就会返回true 在讲解这个问题之前,先了解自动装箱的原理 2.自动装箱的原理 自动装箱,就 ...

  4. 【Python】PDF文档导出指定章节为TXT

    PDF文档导出指定章节为TXT 需求 要导出3000多个pdf文档的特定章节内容为txt格式(pdf文字可复制). 解决 导出PDF 查了一下Python操作PDF文档的方法,主要是通过3个库,PyP ...

  5. access的多个left outer join连接

    虽然你有一万个理由,但是选择ACCESS数据库就是一个不能再蠢的决定. 从AC990账务系统中采集凭证记录,需要做多个左连接.在sql server管理器中执行蛮好,没有问题.可是在程序中就一堆Err ...

  6. 聊聊AI Agent与AI 数字分身

    提供AI应用咨询+陪跑服务,有需要回复1 Manus爆火后,网上出现了很多AI热门名词,比如Agent.AI分身,并且有一张技术架构实现图: 怎么说呢,也许这张图是对的,但就我这边实际的项目实践情况以 ...

  7. 重生之我是操作系统(五)----CPU调度

    简介 当CPU有大量任务要处理,但由于资源有限,无法同时处理.所有就需要某种规则来决定任务处理的顺序,这就是调度. 调度层次 根据调度频率与层次,共分为三种 高级调度 也称为作业调度(Long-Tre ...

  8. 使用Python解决Logistic方程

    引言 在数学和计算机科学中,Logistic 方程是描述人口增长.传播过程等现象的一种常见模型.它通常用于表示一种有限资源下的增长过程,比如动物种群.疾病传播等.本文将带领大家通过 Python 实现 ...

  9. LLM中的Top-K/Top-p/温度都是怎么发挥作用的?

    写在前面 许多大模型具有推理参数,用于控制输出的"随机性".常见的几个是 Top-K.Top-p,以及温度.比如我们常用的 Dify 平台就支持 Top-p 和 温度 的设置: 鼠 ...

  10. nodejs获取一个可用的端口,检查端口是否被占用(完美方案)

    nodejs检查端口是否被占用,先看个运行效果: E:\wamp64\www\tmpPro\tryuseport>node t.js 端口:8022被占用 端口:8023被占用 端口:8024可 ...