先说下当前我为处理动画的思路: (新手上路, 老司机轻喷,如果有更好的实现方法请大神指教 感恩戴德)
#1. 分析动画构成
#2. 如果是位移动画则考虑使用BasicAnimation或者KeyframeAnimation实现, 需要的话再搭配缓动函数
#3. 比较复杂的动画则考虑是否用UIBezierpath一帧帧来画
今天我们模仿做一个场景切换加载等待动画, 比如这样的
我们分析下这张图的构成
#1. 一个灰色的背景
#2. 一个白色的圆环
#3. 一个闭合的圆弧(白色部分)
看起来不是简单的位移动画了, 我们用UIBezierPath加CADisplayLink一帧一帧来画试试看
灰色的背景,
这个比较简单, 我们直接创建一个UIView子类, 背景颜色设置为灰色
白色的圆环,
可以用UIBezierPath直接画一个圆,注意调整线的宽度 So easy
//添加外圆
UIBezierPath *apath = [UIBezierPath bezierPath];
apath.lineWidth = 5;
[apath addArcWithCenter:CGPointMake(50, 50) radius:40 startAngle:0 endAngle:2 * M_PI clockwise:YES];
[apath stroke];
闭合的圆弧,
一样用UIBezierPath, 先设置圆心 画一个圆弧然后闭合路径, _count是设置的一个变量, 有Controller中的计时器控制以达到动画的效果
//先画内圆
//设置线条
path.lineWidth = 5;
path.lineCapStyle = kCGLineCapRound;
path.lineJoinStyle = kCGLineJoinRound;
//设置圆心
[path moveToPoint:CGPointMake(_myWidth / 2, _myHeight / 2)];
//设置内切圆弧
[path addArcWithCenter:CGPointMake(_myWidth / 2, _myHeight / 2) radius:38 startAngle: M_PI * 3 / 2 endAngle:M_PI * 3 / 2 + 2 * M_PI / 300 * _count clockwise:YES];
//线路闭合
[path closePath];
[path fill];
要注意调整外圆和内闭合弧的线宽
然后在Controller中创建计时器, 改变_count的值达到动画的效果
上代码:
先创建一个UIView子类,
#import <UIKit/UIKit.h>
@interface MyView : UIView
@property (nonatomic, assign) CGFloat myWidth;
@property (nonatomic, assign) CGFloat myHeight;
@property (nonatomic, assign) NSInteger count;
@end
在drawRect中添加图案
#import "MyView.h"
@implementation MyView
- (id)initWithFrame:(CGRect)frame {
CGRect myFrame = CGRectMake(frame.origin.x, 0, frame.size.width, frame.size.height);
self = [super initWithFrame:myFrame];
if (self) {
self.myWidth = frame.size.width;
self.myHeight = frame.size.height;
}
return self;
}
- (void)drawRect:(CGRect)rect {
// Drawing code
//设置线条颜色
UIColor *color = [UIColor whiteColor];
[color set];
//画内圆
UIBezierPath *path = [UIBezierPath bezierPath];
//设置线条
path.lineWidth = 5;
path.lineCapStyle = kCGLineCapRound;
path.lineJoinStyle = kCGLineJoinRound;
//设置圆心
[path moveToPoint:CGPointMake(_myWidth / 2, _myHeight / 2)];
//设置内切圆弧
[path addArcWithCenter:CGPointMake(_myWidth / 2, _myHeight / 2)
radius:38 startAngle: M_PI * 3 / 2
endAngle:M_PI * 3 / 2 + 2 * M_PI / 300 * _count
clockwise:YES];
//线路闭合
[path closePath];
[path fill];
//添加外圆
UIBezierPath *apath = [UIBezierPath bezierPath];
apath.lineWidth = 5;
[apath addArcWithCenter:CGPointMake(50, 50)
radius:40
startAngle:0
endAngle:2 * M_PI
clockwise:YES];
[apath stroke];
}
@end
在Controller中调用
#import "ViewController.h"
#import "MyView.h"
@interface ViewController ()
@property (nonatomic, strong) MyView *myView;
@property (nonatomic, strong) CADisplayLink *displayLink;
@property (nonatomic, assign) NSInteger count;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
//创建计时器
_displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(countOn)];
_displayLink.paused = YES;
[_displayLink addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSRunLoopCommonModes];
//创建自定义View实例
_myView = [[MyView alloc] initWithFrame:CGRectMake(150, 0, 100, 100)];
_myView.backgroundColor = [UIColor grayColor];
_myView.count = 1;
[self.view addSubview:_myView];
//在这里可以添加额外动画, 设置hud出现的方式
[UIView animateWithDuration:0.5 animations:^{
_myView.frame = CGRectMake(150, 150, 100, 100);
_displayLink.paused = NO;
}];
}
//计时器事件, 修改动画参数
- (void)countOn {
_count++;
_myView.count = _count;
if (_count > 290) {
[_displayLink invalidate];
[_myView removeFromSuperview];
}
//重绘
[_myView setNeedsDisplay];
}
扫码关注腾讯云开发者
领取腾讯云代金券
Copyright © 2013 - 2025 Tencent Cloud. All Rights Reserved. 腾讯云 版权所有
深圳市腾讯计算机系统有限公司 ICP备案/许可证号:粤B2-20090059 深公网安备号 44030502008569
腾讯云计算(北京)有限责任公司 京ICP证150476号 | 京ICP备11018762号 | 京公网安备号11010802020287
Copyright © 2013 - 2025 Tencent Cloud.
All Rights Reserved. 腾讯云 版权所有