iOS中利用CoreAnimation实现一个时间的进度条效果

发布时间 - 2026-01-11 03:03:58    点击率:

在iOS中实现进度条通常都是通过不停的设置progress来完成的,这样的进度条适用于网络加载(上传下载文件、图片等)。但是对于录制视频这样的需求的话,如果是按照每秒来设置进度的话,显得有点麻烦,于是我就想直接用CoreAnimation来按时间做动画,只要设置最大时间,其他的就不用管了,然后在视频暂停与继续录制时,对动画进行暂停和恢复即可。录制视频的效果如下:

你可以在这里下载demo

那么接下来就是如何用CoreAnimation实现一个进度条控件了。

首先呢,让我们创建一个继承自CAShapeLayer的WKProgressBarLayer。

WKProgressBarLayer默认自身的bounds就是整个进度条的大小。

@interface WKProgressBarLayer : CAShapeLayer
@end

 为了方便外部调用,首先在WKProgressBarLayer.h中定义枚举来表明动画的四个状态

typedef NS_ENUM(NSInteger, WKAnimationStatus) {
 WKAnimationStatusIdle,//空闲
 WKAnimationStatusAnimating,//动画中
 WKAnimationStatusPause,//暂停
 WKAnimationStatusComplete//完成
};

 接下来,定义外部调用的动画接口

@interface WKProgressBarLayer : CAShapeLayer
@property (nonatomic, assign, readonly) WKAnimationStatus animatingStatus;//状态
/**
 开始动画
 @param duration 动画最大时长
 */
- (void)beginAnimationWithDuration:(CGFloat)duration;
/**
 暂停
 */
- (void)pauseAnimation;
/**
 恢复
 */
- (void)resumeAnimation;
/**
 重新开始动画
 @param progress 从哪个进度开始
 @param duration 动画最大时长
 */
- (void)restartAnimationWithProgress:(CGFloat)progress duration:(NSTimeInterval)duration;
@end

 然后,我们在.m实现核心的动画开始方法startAnimtionWithBeginProgress:duration:,详细解释见代码

- (void)startAnimtionWithBeginProgress:(CGFloat)beginProgress duration:(NSTimeInterval)duration
{
 [self reset];//重置动画
 //设置path
 UIBezierPath *fromPath = [UIBezierPath bezierPathWithRect:CGRectMake(0, 0, beginProgress * self.bounds.size.width, self.bounds.size.height)];;
 UIBezierPath *toPath = [UIBezierPath bezierPathWithRect:self.bounds];
 self.path = fromPath.CGPath;
 //创建动画
 CABasicAnimation *animation = [CABasicAnimation animationWithKeyPath:@"path"];
 animation.fromValue = (id)fromPath.CGPath;
 animation.toValue = (id)toPath.CGPath;
 animation.duration = duration;
 [animation setValue:@1 forKey:@"progress"];//用于判断是否是进度动画
 animation.delegate = self; //用于判断动画结束
 [self addAnimation:animation forKey:@"progressAnimation"];
 self.path = toPath.CGPath;
}

 然后呢,需要在动画的delegate与暂停、恢复动画的方法中分别修改动画的状态

- (void)pauseAnimation
{
 CFTimeInterval pausedTime = [self convertTime:CACurrentMediaTime() fromLayer:nil];
 self.speed = 0.0;
 self.timeOffset = pausedTime;
 self.animatingStatus = WKAnimationStatusPause;
}
- (void)resumeAnimation
{
 CFTimeInterval pausedTime = [self timeOffset];
 self.speed = 1.0;
 self.timeOffset = 0.0;
 self.beginTime = 0.0;
 CFTimeInterval timeSincePause = [self convertTime:CACurrentMediaTime() fromLayer:nil] - pausedTime;
 self.beginTime = timeSincePause;
 self.animatingStatus = WKAnimationStatusAnimating;
}
#pragma mark - CAAnimationDelegate
/* Called when the animation begins its active duration. */
- (void)animationDidStart:(CAAnimation *)anim
{
 if (anim == [self animationForKey:@"progressAnimation"]) {//判断进度动画
  self.animatingStatus = WKAnimationStatusAnimating;
 }
}
- (void)animationDidStop:(CAAnimation *)anim finished:(BOOL)flag
{
 if ([anim valueForKey:@"progress"] && flag == YES) {//判断进度动画
  self.animatingStatus = WKAnimationStatusComplete;
 }
}

 至此,进度条layer就完成了,现在创建一个控制器来做测试

首先在storyBoard摆上两个按钮,分别是开始与重置动画(界面搭建很简单,详情见demo)

然后在ViewDidLoad中添加progressLayer

- (void)viewDidLoad {
 [super viewDidLoad];
  
 WKProgressBarLayer *progressLayer = [[WKProgressBarLayer alloc] init];
 progressLayer.frame = CGRectMake(100, 100, 200, 10);
  
 [self.view.layer addSublayer:progressLayer];
  
 self.progressLayer = progressLayer;
}

 接下来,就是动画开始与重置响应

- (IBAction)startOrPauseAction:(UIButton *)sender {
  switch (self.progressLayer.animatingStatus) {
   case WKAnimationStatusIdle:{
    [self.progressLayer beginAnimationWithDuration:10];
   }
    break;
   case WKAnimationStatusAnimating:{
    [self.progressLayer pauseAnimation];
   }
    break;
   case WKAnimationStatusPause:{
    [self.progressLayer resumeAnimation];
   }
    break;
   case WKAnimationStatusComplete:{
    [self.progressLayer restartAnimationWithProgress:0 duration:10];
   }
    break;
   default:
    break;
 }
 sender.selected = !sender.selected;
}
- (IBAction)resetAction:(UIButton *)sender {
 [self.progressLayer restartAnimationWithProgress:0 duration:10];
 self.startOrPauseButton.selected = YES;
}

 以上就是代码主体了,接下来,让我们看看效果

你可以在这里下载demo

总结

以上所述是小编给大家介绍的iOS中利用CoreAnimation实现一个时间的进度条,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!


# coreanimation  # 时间  # 进度条  # iOS实现步骤进度条功能实例代码  # 使用axios实现上传图片进度条功能  # ios开发加载webview显示进度条实例  # iOS 进度条、加载、安装动画的简单实现  # Android仿IOS ViewPager滑动进度条  # iOS实现带动画的环形进度条  # iOS快速实现环形渐变进度条  # iOS中使用NSProgress类来创建UI进度条的方法详解  # IOS实现简单的进度条功能  # iOS中WKWebView仿微信加载进度条  # 在这里  # 你可以  # 让我们  # 创建一个  # 小编  # 时长  # 在此  # 其他的  # 适用于  # 给大家  # 很简单  # 来做  # 来完成  # 所述  # 给我留言  # 如何用  # 感谢大家  # 我就想  # 摆上 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 非常酷的网站设计制作软件,酷培ai教育官方网站?  Chrome浏览器标签页分组怎么用_谷歌浏览器整理标签页技巧【效率】  网站制作报价单模板图片,小松挖机官方网站报价?  Laravel如何使用查询构建器?(Query Builder高级用法)  矢量图网站制作软件,用千图网的一张矢量图做公司app首页,该网站并未说明版权等问题,这样做算不算侵权?应该如何解决?  软银砸40亿美元收购DigitalBridge 强化AI资料中心布局  如何快速生成ASP一键建站模板并优化安全性?  Laravel Admin后台管理框架推荐_Laravel快速开发后台工具  Laravel怎么做缓存_Laravel Cache系统提升应用速度的策略与技巧  如何快速查询网站的真实建站时间?  企业网站制作这些问题要关注  用yum安装MySQLdb模块的步骤方法  php json中文编码为null的解决办法  微信小程序 canvas开发实例及注意事项  如何在IIS管理器中快速创建并配置网站?  Win11怎么设置默认图片查看器_Windows11照片应用关联设置  PythonWeb开发入门教程_Flask快速构建Web应用  如何在阿里云完成域名注册与建站?  nginx修改上传文件大小限制的方法  Laravel如何实现多表关联模型定义_Laravel多对多关系及中间表数据存取【方法】  如何在Windows服务器上快速搭建网站?  Laravel如何实现多级无限分类_Laravel递归模型关联与树状数据输出【方法】  如何用西部建站助手快速创建专业网站?  详解Oracle修改字段类型方法总结  html5源代码发行怎么设置权限_访问权限控制方法与实践【指南】  Swift中swift中的switch 语句  JS中对数组元素进行增删改移的方法总结  Win11怎么查看显卡温度 Win11任务管理器查看GPU温度【技巧】  Python数据仓库与ETL构建实战_Airflow调度流程详解  如何在云主机上快速搭建网站?  Linux后台任务运行方法_nohup与&使用技巧【技巧】  Laravel API资源(Resource)怎么用_格式化Laravel API响应的最佳实践  Laravel观察者模式如何使用_Laravel Model Observer配置  Laravel如何使用withoutEvents方法临时禁用模型事件  Win11怎么恢复误删照片_Win11数据恢复工具使用【推荐】  Laravel任务队列怎么用_Laravel Queues异步处理任务提升应用性能  Laravel怎么处理异常_Laravel自定义异常处理与错误页面教程  重庆市网站制作公司,重庆招聘网站哪个好?  电商网站制作多少钱一个,电子商务公司的网站制作费用计入什么科目?  php在windows下怎么调试_phpwindows环境调试操作说明【操作】  PHP怎么接收前端传的文件路径_处理文件路径参数接收方法【汇总】  用v-html解决Vue.js渲染中html标签不被解析的问题  jQuery validate插件功能与用法详解  Python自动化办公教程_ExcelWordPDF批量处理案例  Laravel怎么配置不同环境的数据库_Laravel本地测试与生产环境动态切换【方法】  Angular 表单中正确绑定输入值以确保提交与验证正常工作  Laravel的Blade指令怎么自定义_创建你自己的Laravel Blade Directives  Laravel全局作用域是什么_Laravel Eloquent Global Scopes应用指南  武汉网站设计制作公司,武汉有哪些比较大的同城网站或论坛,就是里面都是武汉人的?  如何制作新型网站程序文件,新型止水鱼鳞网要拆除吗?