本文实例为大家分享了iOS实现逐帧动画做loading视图的具体代码,供大家参考,具体内容如下
我封装了一个可复用的loading视图组件,用于按照一定周期逐帧播放加载动画。代码如下:
.h文件
?
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
|
#import <UIKit/UIKit.h>
//加载状态
typedef enum {
FZImageSequenceLoadingStatusStop = 1, // 停止
FZImageSequenceLoadingStatusLoading, // 加载中
FZImageSequenceLoadingStatusError //发生错误
} FZImageSequenceLoadingStatus;
@interface FZImageSequenceLoadingView : UIView {
UIImageView *_imageView;
UILabel *_lblMsg;
NSTimer *timer;
int currentImageIndex;
}
@property(strong) NSArray *imageArray; //动画序列的图片数组
@property(strong, nonatomic) UIImage *errorImage;
@property(nonatomic, strong) NSString *errorMsg;
@property(nonatomic, strong) NSString *loadingMsg; //提示文字
@property(nonatomic) CGRect imageFrame; //图片的Frame
@property(nonatomic) CGRect msgFrame; //文字内容的Frame
@property(nonatomic) float timerInterval; //切换图片的周期
/**
切换状态
*/
- (void)switchToStatus:(FZImageSequenceLoadingStatus)status;
/**
通过图片名字和数量设置图片数组,如给定名字"name"、“.png”和数量4,则会去加载“name_1.png”到"name_4.png"的图片
*/
- (void)setImageArrayByName:(NSString *)name andExtName:(NSString *)extName andCount:(int)count;
@end
|
.m文件
?
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
|
#import "FZImageSequenceLoadingView.h"
@implementation FZImageSequenceLoadingView
@synthesize errorImage;
@synthesize errorMsg;
@synthesize imageArray;
@synthesize loadingMsg;
@synthesize timerInterval;
- (id)initWithFrame:(CGRect)frame
{
self = [super initWithFrame:frame];
if (self) {
timerInterval = 1;
currentImageIndex = -1;
}
return self;
}
/*
// Only override drawRect: if you perform custom drawing.
// An empty implementation adversely affects performance during animation.
- (void)drawRect:(CGRect)rect
{
// Drawing code
}
*/
- (void)setupSubviews {
// self.backgroundColor = [UIColor redColor];
if (self.imageArray && self.imageArray.count > 0) {
if (!_imageView) {
_imageView = [[UIImageView alloc] init];
[self addSubview:_imageView];
}
//让图片view为本view的顶部居中,大小为图片数组中的第一张
UIImage *firstImg = [self.imageArray objectAtIndex:0];
_imageView.size = firstImg.size;
_imageView.top = 0;
_imageView.left = (self.size.width - _imageView.size.width) / 2;
}
if (self.loadingMsg) {
CGSize labelSize = [self.loadingMsg sizeWithFont:[UIFont systemFontOfSize:11]];
if (!_lblMsg) {
_lblMsg = [[UILabel alloc] initWithFrame:CGRectZero];
_lblMsg.textAlignment = NSTextAlignmentCenter;
[self addSubview:_lblMsg];
}
_lblMsg.font = [UIFont systemFontOfSize:11];
_lblMsg.size = labelSize;
_lblMsg.textColor = [UIColor darkGrayColor];
_lblMsg.backgroundColor = [UIColor clearColor];
_lblMsg.bottom = self.height;
_lblMsg.left = (self.width - _lblMsg.width) / 2;
}
}
- (void)switchToStatus:(FZImageSequenceLoadingStatus)status {
if (!_lblMsg || !_imageView) {
[self setupSubviews];
}
switch (status) {
case FZImageSequenceLoadingStatusError:
[self switchToError];
break;
case FZImageSequenceLoadingStatusLoading:
[self switchToLoading];
break;
case FZImageSequenceLoadingStatusStop:
[self switchToStop];
break;
}
}
- (void)switchToStop {
[timer invalidate];
timer = nil;
if (self.imageArray && self.imageArray.count > 0) {
_imageView.image = [self.imageArray objectAtIndex:0];
}
}
- (void)switchToError {
[timer invalidate];
timer = nil;
//如果有错误状态的图
if (self.errorImage) {
_imageView.image = self.errorImage;
//如果没有就用第一张动画图
} else if (self.imageArray && self.imageArray.count > 0) {
_imageView.image = [self.imageArray objectAtIndex:0];
}
if (self.errorMsg) {
_lblMsg.text = self.errorMsg;
}
}
- (void)switchToLoading {
if (self.loadingMsg) {
_lblMsg.text = self.loadingMsg;
}
if (!timer) {
timer = [NSTimer scheduledTimerWithTimeInterval:self.timerInterval target:self selector:@selector(showNextImage) userInfo:nil repeats:YES];
}
}
- (void)showNextImage {
if (!imageArray || imageArray.count < 1) {
return;
}
currentImageIndex = (currentImageIndex + 1) % self.imageArray.count;
// 主线程执行:
dispatch_async(dispatch_get_main_queue(), ^{
_imageView.image = [imageArray objectAtIndex:currentImageIndex];
});
}
- (void)setImageArrayByName:(NSString *)name andExtName:(NSString *)extName andCount:(int)count {
NSAssert((name && extName && (count > 0)), @"图片名字和数量错误");
NSMutableArray *imgs = [NSMutableArray arrayWithCapacity:count];
for (int i = 1; i <= count; i++) {
NSString *imgName = [NSString stringWithFormat:@"%@_%i%@", name, i, extName];
UIImage *image = [UIImage imageNamed:imgName];
NSLog(@"%@", image);
if (!image) {
continue;
}
[imgs addObject:image];
}
self.imageArray = imgs;
}
@end
|
使用示例,在uiwebview中使用如下:
初始化视图:
?
|
1
2
3
4
5
6
7
8
9
10
11
12
13
|
//设置loading视图
- (void)setupLoadingView {
if (!_loadingView) {
_loadingView = [[FZImageSequenceLoadingView alloc] initWithFrame:CGRectMake(0, 0, 170, 70)];
_loadingView.center = self.view.center;
[_loadingView setImageArrayByName:@"loading" andExtName:@".png" andCount:10];
_loadingView.loadingMsg = @"努力加载中,请稍候";
_loadingView.errorMsg = @"加载失败";
_loadingView.timerInterval = 0.1;
_loadingView.hidden = YES;
[self.view addSubview:_loadingView];
}
}
|
在uiwebview的代理方法中切换状态:
?
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
#pragma mark - webview delegate
- (void)webViewDidStartLoad:(UIWebView *)webView {
if (_loadingView.hidden) {
_loadingView.hidden = NO;
[_loadingView switchToStatus:FZImageSequenceLoadingStatusLoading];
}
}
- (void)webViewDidFinishLoad:(UIWebView *)webView {
if (!_loadingView.hidden) {
[_loadingView switchToStatus:FZImageSequenceLoadingStatusStop];
_loadingView.hidden = YES;
}
}
- (void)webView:(UIWebView *)webView didFailLoadWithError:(NSError *)error {
NSLog(@"load page error:%@", [error description]);
if (!_loadingView.hidden) {
[_loadingView switchToStatus:FZImageSequenceLoadingStatusError];
}
}
|
目前该组件功能还不够完善,但是能满足目前我自己的需求,后续再继续丰富。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持快网idc。
相关文章
猜你喜欢
- ASP.NET本地开发时常见的配置错误及解决方法? 2025-06-10
- ASP.NET自助建站系统的数据库备份与恢复操作指南 2025-06-10
- 个人网站服务器域名解析设置指南:从购买到绑定全流程 2025-06-10
- 个人网站搭建:如何挑选具有弹性扩展能力的服务器? 2025-06-10
- 个人服务器网站搭建:如何选择适合自己的建站程序或框架? 2025-06-10
TA的动态
- 2025-07-10 怎样使用阿里云的安全工具进行服务器漏洞扫描和修复?
- 2025-07-10 怎样使用命令行工具优化Linux云服务器的Ping性能?
- 2025-07-10 怎样使用Xshell连接华为云服务器,实现高效远程管理?
- 2025-07-10 怎样利用云服务器D盘搭建稳定、高效的网站托管环境?
- 2025-07-10 怎样使用阿里云的安全组功能来增强服务器防火墙的安全性?
快网idc优惠网
QQ交流群
您的支持,是我们最大的动力!
热门文章
-
2025-05-29 56
-
2025-05-29 73
-
2025-05-25 87
-
2025-06-04 70
-
Linux 5.15将修复此前的补丁让软盘驱动器设备出错的问题
2025-05-27 29
热门评论

