iOS 进度条、加载、安装动画的简单实现

首先看一下效果图:

下面贴上代码:

控制器ViewController:

#import <UIKit/UIKit.h>
@interface ViewController : UIViewController
@end
/*** ---------------分割线--------------- ***/
#import "ViewController.h"
#import "HWWaveView.h"
#import "HWCircleView.h"
#import "HWProgressView.h"
#import "HWInstallView.h"
@interface ViewController ()
@property (nonatomic, strong) NSTimer *timer;
@property (nonatomic, weak) HWWaveView *waveView;
@property (nonatomic, weak) HWCircleView *circleView;
@property (nonatomic, weak) HWProgressView *progressView;
@property (nonatomic, weak) HWInstallView *installView;
@end
@implementation ViewController
- (void)viewDidLoad {
 [super viewDidLoad];
 //创建控件
 [self creatControl];
 //添加定时器
 [self addTimer];
}
- (void)creatControl
{
 //波浪
 HWWaveView *waveView = [[HWWaveView alloc] initWithFrame:CGRectMake(30, 100, 150, 150)];
 [self.view addSubview:waveView];
 self.waveView = waveView;
 //圆圈
 HWCircleView *circleView = [[HWCircleView alloc] initWithFrame:CGRectMake(220, 100, 150, 150)];
 [self.view addSubview:circleView];
 self.circleView = circleView;
 //进度条
 HWProgressView *progressView = [[HWProgressView alloc] initWithFrame:CGRectMake(30, 365, 150, 20)];
 [self.view addSubview:progressView];
 self.progressView = progressView;
 //加载安装效果
 HWInstallView *installView = [[HWInstallView alloc] initWithFrame:CGRectMake(220, 300, 150, 150)];
 [self.view addSubview:installView];
 self.installView = installView;
}
- (void)addTimer
{
 _timer = [NSTimer scheduledTimerWithTimeInterval:0.2f target:self selector:@selector(timerAction) userInfo:nil repeats:YES];
 [[NSRunLoop mainRunLoop] addTimer:_timer forMode:NSRunLoopCommonModes];
}
- (void)timerAction
{
 _waveView.progress += 0.01;
 _circleView.progress += 0.01;
 _progressView.progress += 0.01;
 _installView.progress += 0.01;
 if (_waveView.progress >= 1) {
  [self removeTimer];
  NSLog(@"完成");
 }
}
- (void)removeTimer
{
 [_timer invalidate];
 _timer = nil;
}
@end
波浪HWWaveView:
[objc] view plain copy 在CODE上查看代码片派生到我的代码片
#import <UIKit/UIKit.h>
@interface HWWaveView : UIView
@property (nonatomic, assign) CGFloat progress;
@end
/*** ---------------分割线--------------- ***/
#import "HWWaveView.h"
#define KHWWaveFillColor [UIColor groupTableViewBackgroundColor] //填充颜色
#define KHWWaveTopColor [UIColor colorWithRed:0/255.0 green:191/255.0 blue:255/255.0 alpha:1.0f] //前面波浪颜色
#define KHWWaveBottomColor [UIColor colorWithRed:0/255.0 green:191/255.0 blue:255/255.0 alpha:0.4f] //后面波浪颜色
@interface HWWaveView ()
@property (nonatomic, strong) CADisplayLink *displayLink;
@property (nonatomic, assign) CGFloat wave_amplitude;//振幅a(y = asin(wx+φ) + k)
@property (nonatomic, assign) CGFloat wave_cycle;//周期w
@property (nonatomic, assign) CGFloat wave_h_distance;//两个波水平之间偏移
@property (nonatomic, assign) CGFloat wave_v_distance;//两个波竖直之间偏移
@property (nonatomic, assign) CGFloat wave_scale;//水波速率
@property (nonatomic, assign) CGFloat wave_offsety;//波峰所在位置的y坐标
@property (nonatomic, assign) CGFloat wave_move_width;//移动的距离,配合速率设置
@property (nonatomic, assign) CGFloat wave_offsetx;//偏移
@property (nonatomic, assign) CGFloat offsety_scale;//上升的速度
@end
@implementation HWWaveView
- (instancetype)initWithFrame:(CGRect)frame
{
 if (self = [super initWithFrame:frame]) {
  self.backgroundColor = [UIColor clearColor];
  //初始化信息
  [self initInfo];
 }
 return self;
}
- (void)initInfo
{
 //进度
 _progress = 0;
 //振幅
 _wave_amplitude = self.frame.size.height / 25;
 //周期
 _wave_cycle = 22 * M_PI / (self.frame.size.width * 0.9);
 //两个波水平之间偏移
 _wave_h_distance = 22 * M_PI / _wave_cycle * 0.6;
 //两个波竖直之间偏移
 _wave_v_distance = _wave_amplitude * 0.4;
 //移动的距离,配合速率设置
 _wave_move_width = 0.5;
 //水波速率
 _wave_scale = 0.4;
 //上升的速度
 _offsety_scale = 0.1;
 //波峰所在位置的y坐标,刚开始的时候_wave_offsety是最大值
 _wave_offsety = (1 - _progress) * (self.frame.size.height + 22 * _wave_amplitude);
 [self addDisplayLinkAction];
}
- (void)addDisplayLinkAction
{
 _displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(displayLinkAction)];
 [_displayLink addToRunLoop:[NSRunLoop mainRunLoop] forMode:NSRunLoopCommonModes];
}
- (void)displayLinkAction
{
 _wave_offsetx += _wave_move_width * _wave_scale;
 //完成
 if (_wave_offsety <= 0.01) [self removeDisplayLinkAction];
 [self setNeedsDisplay];
}
- (void)removeDisplayLinkAction
{
 [_displayLink invalidate];
 _displayLink = nil;
}
- (void)drawRect:(CGRect)rect
{
 UIBezierPath *path = [UIBezierPath bezierPathWithOvalInRect:rect];
 [KHWWaveFillColor setFill];
 [path fill];
 [path addClip];
 //绘制两个波形图
 [self drawWaveColor:KHWWaveTopColor offsetx:0 offsety:0];
 [self drawWaveColor:KHWWaveBottomColor offsetx:_wave_h_distance offsety:_wave_v_distance];
}
- (void)drawWaveColor:(UIColor *)color offsetx:(CGFloat)offsetx offsety:(CGFloat)offsety
{
 //波浪动画,进度的实际操作范围是,多加上两个振幅的高度,到达设置进度的位置y
 CGFloat end_offY = (1 - _progress) * (self.frame.size.height + 22 * _wave_amplitude);
 if (_wave_offsety != end_offY) {
  if (end_offY < _wave_offsety) {
   _wave_offsety = MAX(_wave_offsety -= (_wave_offsety - end_offY) * _offsety_scale, end_offY);
  }else {
   _wave_offsety = MIN(_wave_offsety += (end_offY - _wave_offsety) * _offsety_scale, end_offY);
  }
 }
 UIBezierPath *wavePath = [UIBezierPath bezierPath];
 for (float next_x = 0.f; next_x <= self.frame.size.width; next_x ++) {
  //正弦函数,绘制波形
  CGFloat next_y = _wave_amplitude * sin(_wave_cycle * next_x + _wave_offsetx + offsetx / self.bounds.size.width * 22 * M_PI) + _wave_offsety + offsety;
  if (next_x == 0) {
   [wavePath moveToPoint:CGPointMake(next_x, next_y - _wave_amplitude)];
  }else {
   [wavePath addLineToPoint:CGPointMake(next_x, next_y - _wave_amplitude)];
  }
 }
 [wavePath addLineToPoint:CGPointMake(self.frame.size.width, self.frame.size.height)];
 [wavePath addLineToPoint:CGPointMake(0, self.bounds.size.height)];
 [color set];
 [wavePath fill];
}
@end
圆圈HWCircleView:
[objc] view plain copy 在CODE上查看代码片派生到我的代码片
#import <UIKit/UIKit.h>
@interface HWCircleView : UIView
@property (nonatomic, assign) CGFloat progress;
@end
/*** ---------------分割线--------------- ***/
#import "HWCircleView.h"
#define KHWCircleLineWidth 10.0f
#define KHWCircleFont [UIFont boldSystemFontOfSize:26.0f]
#define KHWCircleColor [UIColor colorWithRed:0/255.0 green:191/255.0 blue:255/255.0 alpha:1]
@interface HWCircleView ()
@property (nonatomic, weak) UILabel *cLabel;
@end
@implementation HWCircleView
- (instancetype)initWithFrame:(CGRect)frame
{
 if (self = [super initWithFrame:frame]) {
  self.backgroundColor = [UIColor clearColor];
  //百分比标签
  UILabel *cLabel = [[UILabel alloc] initWithFrame:self.bounds];
  cLabel.font = KHWCircleFont;
  cLabel.textColor = KHWCircleColor;
  cLabel.textAlignment = NSTextAlignmentCenter;
  [self addSubview:cLabel];
  self.cLabel = cLabel;
 }
 return self;
}
- (void)setProgress:(CGFloat)progress
{
 _progress = progress;
 _cLabel.text = [NSString stringWithFormat:@"%d%%", (int)floor(progress * 100)];
 [self setNeedsDisplay];
}
- (void)drawRect:(CGRect)rect
{
 //路径
 UIBezierPath *path = [[UIBezierPath alloc] init];
 //线宽
 path.lineWidth = KHWCircleLineWidth;
 //颜色
 [KHWCircleColor set];
 //拐角
 path.lineCapStyle = kCGLineCapRound;
 path.lineJoinStyle = kCGLineJoinRound;
 //半径
 CGFloat radius = (MIN(rect.size.width, rect.size.height) - KHWCircleLineWidth) * 0.5;
 //画弧(参数:中心、半径、起始角度(3点钟方向为0)、结束角度、是否顺时针)
 [path addArcWithCenter:(CGPoint){rect.size.width * 0.5, rect.size.height * 0.5} radius:radius startAngle:M_PI * 1.5 endAngle:M_PI * 1.5 + M_PI * 22 * _progress clockwise:YES];
 //连线
 [path stroke];
}
@end
进度条HWProgressView:
[objc] view plain copy 在CODE上查看代码片派生到我的代码片
#import <UIKit/UIKit.h>
@interface HWProgressView : UIView
@property (nonatomic, assign) CGFloat progress;
@end
/*** ---------------分割线--------------- ***/
#import "HWProgressView.h"
#define KProgressBorderWidth 2.0f
#define KProgressPadding 1.0f
#define KProgressColor [UIColor colorWithRed:0/255.0 green:191/255.0 blue:255/255.0 alpha:1]
@interface HWProgressView ()
@property (nonatomic, weak) UIView *tView;
@end
@implementation HWProgressView
- (instancetype)initWithFrame:(CGRect)frame
{
 if (self = [super initWithFrame:frame]) {
  //边框
  UIView *borderView = [[UIView alloc] initWithFrame:self.bounds];
  borderView.layer.cornerRadius = self.bounds.size.height * 0.5;
  borderView.layer.masksToBounds = YES;
  borderView.backgroundColor = [UIColor whiteColor];
  borderView.layer.borderColor = [KProgressColor CGColor];
  borderView.layer.borderWidth = KProgressBorderWidth;
  [self addSubview:borderView];
  //进度
  UIView *tView = [[UIView alloc] init];
  tView.backgroundColor = KProgressColor;
  tView.layer.cornerRadius = (self.bounds.size.height - (KProgressBorderWidth + KProgressPadding) * 2) * 0.5;
  tView.layer.masksToBounds = YES;
  [self addSubview:tView];
  self.tView = tView;
 }
 return self;
}
- (void)setProgress:(CGFloat)progress
{
 _progress = progress;
 CGFloat margin = KProgressBorderWidth + KProgressPadding;
 CGFloat maxWidth = self.bounds.size.width - margin * 2;
 CGFloat heigth = self.bounds.size.height - margin * 2;
 _tView.frame = CGRectMake(margin, margin, maxWidth * progress, heigth);
}
@end
加载安装效果HWInstallView:
[objc] view plain copy 在CODE上查看代码片派生到我的代码片
#import <UIKit/UIKit.h>
@interface HWInstallView : UIView
@property (nonatomic, assign) CGFloat progress;
@end
/*** ---------------分割线--------------- ***/
#import "HWInstallView.h"
#define KHWInstallViewMargin 10
#define KHWInstallColor [UIColor colorWithRed:0/255.0 green:191/255.0 blue:255/255.0 alpha:1]
@implementation HWInstallView
- (instancetype)initWithFrame:(CGRect)frame
{
 if (self = [super initWithFrame:frame]) {
  self.backgroundColor = [UIColor clearColor];
 }
 return self;
}
- (void)setProgress:(CGFloat)progress
{
 _progress = progress;
 [self setNeedsDisplay];
}
- (void)drawRect:(CGRect)rect
{
 CGContextRef context = UIGraphicsGetCurrentContext();
 CGFloat xCenter = rect.size.width * 0.5;
 CGFloat yCenter = rect.size.height * 0.5;
 CGFloat radius = MIN(rect.size.width, rect.size.height) * 0.5 - KHWInstallViewMargin;
 //背景遮罩
 [KHWInstallColor set];
 CGFloat lineW = MAX(rect.size.width, rect.size.height) * 0.5;
 CGContextSetLineWidth(context, lineW);
 CGContextAddArc(context, xCenter, yCenter, radius + lineW * 0.5 + 5, 0, M_PI * 2, 1);
 CGContextStrokePath(context);
 //进程圆
 CGContextSetLineWidth(context, 1);
 CGContextMoveToPoint(context, xCenter, yCenter);
 CGContextAddLineToPoint(context, xCenter, 0);
 CGFloat endAngle = - M_PI * 0.5 + _progress * M_PI * 2 + 0.001;
 CGContextAddArc(context, xCenter, yCenter, radius, - M_PI * 0.5, endAngle, 1);
 CGContextFillPath(context);
}
@end 

以上所述是小编给大家介绍的iOS 进度条、加载、安装动画的简单实现,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

时间: 2017-03-05

iOS动画解析之圆球加载动画XLBallLoading的实现

前言 当网页的页面大小较大,用户加载可能需要较长的时间,在这些情况下,我们一般会用到(加载)loading动画,提示于用户页面在加载中,本文将详细给大家介绍关于iOS圆球加载动画XLBallLoading实现的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧. 一.显示效果 二.原理分析 1.拆解动画 从效果图来看,动画可拆解成两部分:放大动画.位移动画 放大动画 比较简单,这里主要来分析一下位移动画 (1).先去掉缩放效果: 屏蔽放大效果 (2).去掉其中的一个圆球 现

iOS 实现简单的加载等待动画示例(思路与实现)

先看下最后基本要实现的效果 总结一下自己的实现思路与所用到的类 1.这个肯定是要自定义的View类,起名为XDColorCircle吧,最后用的时候达到这样的效果 //创建XDColorCircle的实例化对象 XDColorCircle *circle=[[XDColorCircle alloc]initWithFrame:CGRectMake(0 ,100,self.view.frame.size.width,200)]; //添加到视图上展示 [self.view addSubview:c

iOS利用CALayer实现动画加载的效果

首先来看看效果图 实现过程如下 控制器调用就一句代码: [self showLoadingInView:self.view]; 方便控制器如此调用,就要为控制器添加一个分类 .h文件 #import <UIKit/UIKit.h> #import "GQCircleLoadView.h" @interface UIViewController (GQCircleLoad) //显示动画 - (void)showLoadingInView:(UIView*)view; //隐

iOS 仿微博客户端红包加载界面 XLDotLoading效果

一.显示效果 二.原理简介 1.思路 要实现这个效果需要先知道这两个硬币是怎样运动的,然后通过放大.缩小的效果实现的这种有距离感的效果.思路如下: 一.这两个硬币是在一定范围内做相对运动的,可以先使一个硬币在一个固定范围内做左右的往复运动,另一个硬币和它做"相对运动"即可. 二.让硬币从左至右移动时先变小再回复正常:从右至左移动时先变大再回复正常:这样就实现了这用有距离感的"相对运动". 2.代码 第一步 要实现一个硬币在一定范围内实现左右往复运动,需要先固定一个范

微信JSSDK多图片上传并且解决IOS系统上传一直加载的问题

微信多图片上传必须挨个上传,也就是不能并行,得串行: 那么我们可以定义一个如下所示的上传函数: var serverIds = []; function uploadImages(localImagesIds) { if (localImagesIds.length === 0) { $.showPreloader('正在提交数据...'); $('form').submit(); } wx.uploadImage({ localId: localImagesIds[0], // 需要上传的图片

iOS开发中用imageIO渐进加载图片及获取exif的方法

imageIO完成渐进加载图片 一.常见渐进加载图片模式   目前我们看到的渐进加载主要有以下三种实现方式:   1)  依次从web上加载不同尺寸的图片,从小到大.最开始先拉取一个小缩略图做拉伸显示,然后拉取中等规格的图,拉取完毕直接覆盖显示,最后拉取原图,拉取完成后显示原图.   2)直接从web上拉取最大的图片,每接受一点儿数据就显示一点儿图片,这样就会实现从上到下一点点刷新出来的效果.   3)结合第1种和第2种,先拉取一个缩略图做拉伸显示,然后采用第二种方法直接拉取原图,这样即可以实现

Android自定义View实现loading动画加载效果

项目开发中对Loading的处理是比较常见的,安卓系统提供的不太美观,引入第三发又太麻烦,这时候自己定义View来实现这个效果,并且进行封装抽取给项目提供统一的loading样式是最好的解决方式了. 先自定义一个View,继承自LinearLayout,在Layout中,添加布局控件 /** * Created by xiedong on 2017/3/7. */ public class Loading_view extends LinearLayout { private Context m

iOS开发中UIWebView的加载本地数据的三种方式

UIWebView是IOS内置的浏览器,可以浏览网页,打开文档 html/htm pdf docx txt等格式的文件. safari浏览器就是通过UIWebView做的. 服务器将MIME的标识符等放入传送的数据中告诉浏览器使用那种插件读取相关文件. uiwebview加载各种本地文件(通过loadData方法): UIWebView加载内容的三种方式: 1 加载本地数据文件 指定文件的MIMEType 编码格式使用@"UTF-8" 2加载html字符串(可以加载全部或者部分html

利用JS实现简单的瀑布流加载图片效果

今天学习了一个瀑布流加载效果,很多网站都有瀑布流效果,瀑布流就是很多产品显示在网页上,宽相同,高度不同,表现为多栏布局,随着页面滚动条向下滚动,这种布局还会不断加载数据块并附加至当前尾部. 原理是: 1.设定一行中的列数: 2.取第一行中每一个div的高度并把每一个高度放进一个数组中: 3.算出数组中最小高度的index值: 4.把第二行的第一个div放到最小高度的div的下方并把重新算出的高度值放进数组中,重新计算最小高度的index值: 5.以此类推实现多栏布局的瀑布流效果: 6.如果最后一

Android使用View Animation实现动画加载界面

之前分别介绍了View Animation和Drawable Animation,学了就要用啊,今天给大家一个使用View Animation实现动画加载界面的实现. 首先先看一下实现效果. 下面是实现代码 package com.example.animationloading; import java.util.Timer; import java.util.TimerTask; import android.annotation.SuppressLint; import android.a

Android框架Volley之利用Imageloader和NetWorkImageView加载图片的方法

首先我们在项目中导入这个框架: implementation 'com.mcxiaoke.volley:library:1.0.19' 在AndroidManifest文件当中添加网络权限: <uses-permission android:name="android.permission.INTERNET"/> 下面是我们的首页布局: 在这个布局当中我们将Volley框架的所有功能都做成了一个按钮,按下按钮之后就会在"显示结果"下面显示结果,显示结果下

vue初始化动画加载的实例

1.在入口文件index.html中加入loading动画: <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="renderer" content="webkit|ie-comp|ie-stand"> <meta content="width=device-width, initial-sca