封装了一个iOS对号成功动画

打印 上一主题 下一主题

主题 1009|帖子 1009|积分 3027


基本思绪实在很简单,就是通过贝塞尔曲线画出路径,然后
利用CAShapeLayer 渲染路径,然后通过strokeEnd 动画实现
路径的结果,这里注意,这个过程中过碰到过一个题目,就是
对号动画完成之后,整个对号不见了,后来颠末仔细观察,发现,是自己初始化 checkLayer的时候,将strokeEnd属设置为0了,注意,虽然我们是通过"strokeEnd"设置的动画,但是我们进行动画之后,并不会真正的改变layer.strokeEnd属性的值,所以我们初始化对号layer的时候,照旧要将strokeEnd设置为1
下面贴出所有代码
  1. //
  2. //  ViewController.m
  3. //  LBProgressCircle
  4. //
  5. //  Created by mac on 2024/5/31.
  6. //
  7. #import "ViewController.h"
  8. @interface ViewController ()
  9. @property (nonatomic, strong) UIView *loadingView;
  10. //进度圆环曲线
  11. @property (nonatomic, strong) UIBezierPath *circlePath;
  12. //整个圆环
  13. @property (nonatomic, strong) CAShapeLayer *wholeCircleLayer;
  14. //进度圆环layer
  15. @property (nonatomic, strong) CAShapeLayer *progressLayer;
  16. //对号圆环
  17. @property (nonatomic, strong) CAShapeLayer *checkRoundLayer;
  18. //对号layer
  19. @property (nonatomic, strong) CAShapeLayer *checkLayer;
  20. @end
  21. @implementation ViewController
  22. - (void)viewDidLoad {
  23.     [super viewDidLoad];
  24.    
  25.     self.view.backgroundColor = [UIColor lightGrayColor];
  26.     [self.view addSubview:self.loadingView];
  27.     [self.loadingView.layer addSublayer:self.wholeCircleLayer];
  28.     [self.loadingView.layer addSublayer:self.progressLayer];
  29.     [self.loadingView.layer addSublayer:self.checkLayer];
  30.     [self.loadingView.layer addSublayer:self.checkRoundLayer];
  31.     [self handle];
  32.    
  33.     // Do any additional setup after loading the view.
  34. }
  35. - (void)handle
  36. {
  37.     self.progressLayer.strokeEnd = 0;
  38.     __block CGFloat second = 0;
  39.     NSTimer *time = [NSTimer scheduledTimerWithTimeInterval:0.1 repeats:YES block:^(NSTimer * _Nonnull timer) {
  40.         if (second >= 1) {
  41.             return;
  42.         }
  43.         second += 0.1;
  44.         [self updateProgress:second];
  45.     }];
  46. }
  47. - (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
  48. {
  49.     [self handle];
  50. }
  51. - (void)updateProgress:(CGFloat)progress
  52. {
  53.     self.progressLayer.strokeEnd = progress;
  54.     if (progress >= 1) {
  55.         [self showProgressCirce:NO];
  56.         [self.checkRoundLayer addAnimation:[self animation] forKey:@"strokeEnd"];
  57.         dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.5 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  58.             self.checkLayer.hidden = NO;
  59.             [self.checkLayer addAnimation:[self animation] forKey:@"strokeEnd"];
  60.         });
  61.     } else {
  62.         self.checkLayer.hidden = YES;
  63.         [self showProgressCirce:YES];
  64.     }
  65. }
  66. - (void)showProgressCirce:(BOOL)showCircle
  67. {
  68.     self.checkRoundLayer.hidden = showCircle;
  69.     self.wholeCircleLayer.hidden = !showCircle;
  70.     self.progressLayer.hidden = !showCircle;
  71. }
  72. #pragma mark - lazy load
  73. - (UIView *)loadingView
  74. {
  75.     if (!_loadingView) {
  76.         _loadingView = [[UIView alloc] initWithFrame:CGRectMake(100, 100, 100, 100)];
  77.         _loadingView.backgroundColor = [UIColor blackColor];
  78.     }
  79.     return _loadingView;
  80. }
  81. - (UIBezierPath *)circlePath
  82. {
  83.     if (!_circlePath) {
  84.         _circlePath = [UIBezierPath bezierPathWithArcCenter:CGPointMake(27.5, 44) radius:19 startAngle:-M_PI_2 endAngle:-M_PI_2 + M_PI * 2 clockwise:YES];
  85.         _circlePath.lineCapStyle = kCGLineCapRound;
  86.         _circlePath.lineJoinStyle = kCGLineJoinRound;
  87.     }
  88.     return _circlePath;
  89. }
  90. - (CAShapeLayer *)progressLayer
  91. {
  92.     if (!_progressLayer) {
  93.         _progressLayer = [[CAShapeLayer alloc] init];
  94.         _progressLayer.path = self.circlePath.CGPath;
  95.         _progressLayer.strokeStart = 0;
  96.         _progressLayer.strokeEnd = 0;
  97.         _progressLayer.strokeColor = [UIColor redColor].CGColor;
  98.         _progressLayer.fillColor = [UIColor clearColor].CGColor;
  99.         _progressLayer.lineWidth = 2;
  100.     }
  101.     return _progressLayer;
  102. }
  103. - (CAShapeLayer *)wholeCircleLayer
  104. {
  105.     if (!_wholeCircleLayer) {
  106.         _wholeCircleLayer = [[CAShapeLayer alloc] init];
  107.         _wholeCircleLayer.path = self.circlePath.CGPath;
  108.         _wholeCircleLayer.strokeStart = 0;
  109.         _wholeCircleLayer.strokeEnd = 1;
  110.         _wholeCircleLayer.strokeColor = [[UIColor redColor] colorWithAlphaComponent:0].CGColor;
  111.         _wholeCircleLayer.fillColor = [UIColor clearColor].CGColor;
  112.         _wholeCircleLayer.lineWidth = 2;
  113.     }
  114.     return _wholeCircleLayer;
  115. }
  116. - (CAShapeLayer *)checkRoundLayer
  117. {
  118.     if (!_checkRoundLayer) {
  119.         UIBezierPath *path = [UIBezierPath bezierPathWithArcCenter:CGPointMake(27.5, 44) radius:16 startAngle:- M_PI_2 endAngle:- M_PI_2 + M_PI * 2 clockwise:YES];
  120.         path.lineCapStyle = kCGLineCapRound;
  121.         path.lineJoinStyle = kCGLineJoinRound;
  122.         _checkRoundLayer = [[CAShapeLayer alloc] init];
  123.         _checkRoundLayer.path = path.CGPath;
  124.         _checkRoundLayer.strokeColor = [UIColor whiteColor].CGColor;
  125.         _checkRoundLayer.fillColor = [UIColor clearColor].CGColor;
  126.         _checkRoundLayer.hidden = YES;
  127.         _checkRoundLayer.lineWidth = 2;
  128.     }
  129.     return _checkRoundLayer;
  130. }
  131. - (CAShapeLayer *)checkLayer
  132. {
  133.     if (!_checkLayer) {
  134.         UIBezierPath *path = [[UIBezierPath alloc] init];
  135.         path.lineCapStyle = kCGLineCapRound;
  136.         path.lineJoinStyle = kCGLineJoinRound;
  137.         [path moveToPoint:CGPointMake(19, 43)];
  138.         CGPoint pl = CGPointMake(25, 49);
  139.         [path addLineToPoint:pl];
  140.         CGPoint p2 = CGPointMake(36, 39);
  141.         [path addLineToPoint:p2];
  142.         _checkLayer = [[CAShapeLayer alloc] init];
  143.         _checkLayer.fillColor = [UIColor clearColor].CGColor;
  144.         //线条颜色
  145.         _checkLayer.strokeColor = [UIColor whiteColor].CGColor;
  146.         _checkLayer.lineWidth = 2;
  147.         _checkLayer.path = path.CGPath;
  148.         _checkLayer.hidden = YES;
  149.     }
  150.     return _checkLayer;
  151. }
  152. - (CABasicAnimation *)animation
  153. {
  154.     CABasicAnimation *checkAnimation = [CABasicAnimation animationWithKeyPath:@"strokeEnd"];
  155.     checkAnimation.duration = 0.5;
  156.     checkAnimation.fromValue = @(0);
  157.     checkAnimation.toValue = @(1);
  158.     return checkAnimation;
  159. }
  160. @end
复制代码
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

麻花痒

论坛元老
这个人很懒什么都没写!
快速回复 返回顶部 返回列表