【iOS】画圈的轨迹动画

动画

最近接到一个需求,需要做一个启动的加载动画,如下图:

load.gif

思考下,可以拆分几个部分

动画拆分

  • 1 . 中间圆圈的旋转动画!
  • 2 . 两边的画线轨迹!
  • 3 . 动态变化的进度显示!

1.中间的圆圈旋转

我这边使用的4张图片,已附上:


>
02.png
03.png
juhua.png

是不是很明显,使用图片的好处就是可以是动画颜色和样式更多变!
那么怎么让它变成旋转的呢?其实很简单,Apple 就已经提供给我们很好的API 以供使用!

首先声明需要使用的图片视图:

/**
 加载图
 */
@property (nonatomic,strong) UIImageView * imageView1;
@property (nonatomic,strong) UIImageView * imageView2;
@property (nonatomic,strong) UIImageView * imageView3;
@property (nonatomic,strong) UIImageView * hudImgView;

//旋转角度
@property (nonatomic,assign) CGFloat angle1;
@property (nonatomic,assign) CGFloat angle2;
@property (nonatomic,assign) CGFloat angle3;
@property (nonatomic,assign) CGFloat angle4;

/**
 是否停止加载
 */
@property (nonatomic,assign) BOOL isStopLoad;

接下来UI实现部分:

- (void)loadUI
{
    
    CGPoint point   = self.center;
    CGRect  frame   = CGRectMake(point.x - 60, point.y - 40, 120, 120);
    //背景视图:
    _bgImageView = [[UIImageView alloc]initWithFrame:self.bounds];
    _bgImageView.image = [UIImage imageNamed:@"loading"];
    [self addSubview:_bgImageView];
    
    _imageView1 = [[UIImageView alloc]initWithFrame:frame];
    //    _imageView1.center = self.center;
    _imageView1.image = [UIImage imageNamed:@"01"];
    [self addSubview:_imageView1];
    
    _imageView2 = [[UIImageView alloc]initWithFrame:frame];
    //    _imageView2.center = self.center;
    _imageView2.image = [UIImage imageNamed:@"02"];
    [self addSubview:_imageView2];
    
    _imageView3 = [[UIImageView alloc]initWithFrame:frame];
    //    _imageView3.center = self.center;
    _imageView3.image = [UIImage imageNamed:@"03"];
    [self addSubview:_imageView3];
    
    _hudImgView = [[UIImageView alloc]initWithFrame:frame];
    //    _hudImgView.center = self.center;
    _hudImgView.image = [UIImage imageNamed:@"juhua"];
    [self addSubview:_hudImgView];
   
}

最后一步,也就是关键的旋转动画,在需要开启的地方调用如下这个方法:

ps: 旋转角度和旋转速度可以根据需要自己进行更改!

/**
 旋转动画
 */
- (void)startAnimation
{
    //选择动画:
    CGAffineTransform endAngle1 = CGAffineTransformMakeRotation(_angle1 * (M_PI / 180.0f));
    CGAffineTransform endAngle2 = CGAffineTransformMakeRotation(_angle2 * (M_PI / 180.0f));
    CGAffineTransform endAngle3 = CGAffineTransformMakeRotation(_angle3 * (M_PI / 180.0f));
    CGAffineTransform endAngle4 = CGAffineTransformMakeRotation(_angle4 * (M_PI / 180.0f));
    
    dispatch_async(dispatch_get_main_queue(), ^{
        
        [UIView animateWithDuration:0.01 delay:0 options:UIViewAnimationOptionCurveLinear animations:^{
            
            _imageView1.transform = endAngle1;
            _imageView2.transform = endAngle2;
            _imageView3.transform = endAngle3;
            _hudImgView.transform = endAngle4;
            
        } completion:^(BOOL finished) {
            _angle1 -= 4;
            _angle2 += 4;
            _angle3 -= 4;
            _angle4 += 4;
            
            if (!_isStopLoad) {
                
                [self startAnimation];
            }
            
        }];
        
        
    });
 
}

2.进度动态显示:

这个就不再赘述,直接上代码:

- (void)loadUI
{
 
    //进度显示Lable
    _progressLable = [[UILabel alloc]initWithFrame:CGRectMake(0, 0, 50, 50)];
    _progressLable.center = _hudImgView.center;
    _progressLable.textColor = CenterColor;
    _progressLable.textAlignment = NSTextAlignmentCenter;
    _progressLable.font = [UIFont boldSystemFontOfSize:12];
    _progressLable.layer.cornerRadius = CGRectGetWidth(_progressLable.frame)/2;
    _progressLable.layer.masksToBounds = YES;
    _progressLable.layer.borderColor = CenterColor.CGColor;
    _progressLable.layer.borderWidth = 0.8f;
    _progressLable.text = @"0%";
    [self addSubview:_progressLable];
    
}

关于进度我这里因为Demo原因,使用的假数据,用的随机数!

/**
 进度显示
 */
- (void)startProgressAnimation
{
    if (_isStopLoad == YES) {
        return;
    }
    //随机数
    _progressNumber += rand()%20;
    
    if (_progressNumber >=100)
    {
        _progressNumber = 100;
    }
    _progressLable.text = [NSString stringWithFormat:@"%ld%@",_progressNumber,@"%"];
    
}

最后一个,前言说到的画线轨迹动画!

很明显,动画轨迹来看,一个直线和一个半圆,
既然如此,那么就可以按照这个想法来进行实现!我们都知道,OC中有个很强大的类,UIBezierPath 轨迹画线!
可供参考的 :
UIBezierPath苹果官方API文献
简书李国安的文章

下面就是代码实现部分,先贴代码:

-(void)drawRect:(CGRect)rect
{
    //上半部轨迹:
    UIBezierPath * path = [UIBezierPath bezierPath];
    //起点
    [path moveToPoint:CGPointMake(ScreenWidth, _hudImgView.center.y)];
    //直线终点
    [path addLineToPoint:CGPointMake(self.center.x+60, _hudImgView.center.y)];
    //圆路径
    [path addArcWithCenter:_hudImgView.center radius:60.0f startAngle:M_PI*2 endAngle:M_PI*1 clockwise:NO];
    //创建一个CAShapeLayer
    CAShapeLayer * caShapelayer = [CAShapeLayer layer];
    caShapelayer.path = path.CGPath;
    //线条宽度
    caShapelayer.lineWidth = 1.5f;
    caShapelayer.frame = self.bounds;
    //线条颜色
    caShapelayer.strokeColor = CenterColor.CGColor;
    //填充颜色
    caShapelayer.fillColor = [UIColor clearColor].CGColor;
    [self.layer addSublayer:caShapelayer];
    
    //下半部轨迹:
    UIBezierPath * path2 = [UIBezierPath bezierPath];
    [path2 moveToPoint:CGPointMake(0, _hudImgView.center.y)];
    [path2 addLineToPoint:CGPointMake(self.center.x-60, _hudImgView.center.y)];
    [path2 addArcWithCenter:_hudImgView.center radius:60.0f startAngle:M_PI*1 endAngle:M_PI*2 clockwise:NO];
    //创建一个CAShapeLayer
    CAShapeLayer * caShapelayer2 = [CAShapeLayer layer];
    caShapelayer2.path = path2.CGPath;
    //线条宽度
    caShapelayer2.lineWidth = 1.5f;
    caShapelayer2.frame = self.bounds;
    //线条颜色
    caShapelayer2.strokeColor = CenterColor.CGColor;
    //填充颜色
    caShapelayer2.fillColor = [UIColor clearColor].CGColor;
    [self.layer addSublayer:caShapelayer2];
    
    //开始动画
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        [self drawLineAnimation:caShapelayer];
        [self drawLineAnimation:caShapelayer2];
    });
}
//绘制 Path轨迹
- (void)drawLineAnimation:(CALayer*)layer {
    
    CABasicAnimation *bas = [CABasicAnimation animationWithKeyPath:@"strokeEnd"];
    bas.duration = 2;
    bas.delegate = self;
    bas.fromValue = [NSNumber numberWithInteger:0];
    bas.toValue = [NSNumber numberWithInteger:1];
    [layer addAnimation:bas forKey:@"key"];
    
}

总体来说,上下部分轨迹,实际上是一样的,所以就分析其中一个就可以了,首先,确定path 轨迹起点,拐点,和终点!

 //起点
    [path moveToPoint:CGPointMake(ScreenWidth, _hudImgView.center.y)];
    //直线终点
    [path addLineToPoint:CGPointMake(self.center.x+60, _hudImgView.center.y)];
    //圆路径
    [path addArcWithCenter:_hudImgView.center radius:60.0f startAngle:M_PI*2 endAngle:M_PI*1 clockwise:NO];

直线路径无需多说,注释已经很清晰了,主要说下,半圆轨迹:
如下方法:
center : 顾名思义就是圆的中心点,学过数学的都知道!
radius: 圆的半径!
startAngle: 圆轨迹起始的弧度!
endAngle : 圆轨迹结束的弧度!
clockwise : 是否顺时针!

- (void)addArcWithCenter:(CGPoint)center
                  radius:(CGFloat)radius 
              startAngle:(CGFloat)startAngle 
                endAngle:(CGFloat)endAngle 
               clockwise:(BOOL)clockwise;

轨迹已经给出,那么其余属性见注释,可以尝试自己改动看效果!

至此所有的动画效果都已经完成,在有需要的地方进行调用,就可以了!

ps: 不当之处还请海涵,分享以及记录自己的代码,还请不喜勿喷!!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,126评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,254评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,445评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,185评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,178评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,970评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,276评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,927评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,400评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,883评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,997评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,646评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,213评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,204评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,423评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,423评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,722评论 2 345

推荐阅读更多精彩内容