iOS自定义UIButton点击动画特效

 更新时间:2019年04月28日 14:19:53   作者:hero_wqb  
这篇文章主要为大家详细介绍了iOS自定义UIButton点击动画特效,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

借鉴相关资料,整理了一个很有意思的button动画效果,iOS自定义UIButton点击动画特效

先看一下效果图:

下面贴上代码:

ViewController:

#import <UIKit/UIKit.h>
 
@interface ViewController : UIViewController
 
@end
 
#import "ViewController.h"
#import "HWButton.h"
 
#define mainW [UIScreen mainScreen].bounds.size.width
#define mainH [UIScreen mainScreen].bounds.size.height
 
@interface ViewController ()
 
@end
 
@implementation ViewController
 
- (void)viewDidLoad {
  [super viewDidLoad];
 
  self.view.backgroundColor = [UIColor blackColor];
 
  //创建控件
  [self creatButton];
}
 
- (void)creatButton
{
  HWButton *button = [[HWButton alloc] initWithFrame:CGRectMake(mainW * 0.5 - 60, mainH - 100, 120, 72) maxLeft:100 maxRight:100 maxHeight:300];
  [button setImage:[UIImage imageNamed:@"button"] forState:UIControlStateNormal];
  button.images = @[[UIImage imageNamed:@"Circle 1"], [UIImage imageNamed:@"Circle 2"], [UIImage imageNamed:@"Circle 3"], [UIImage imageNamed:@"Hero"]];
  button.duration = 10;
  [button addTarget:self action:@selector(buttonOnClick:) forControlEvents:UIControlEventTouchUpInside];
  [self.view addSubview:button];
}
 
- (void)buttonOnClick:(HWButton *)btn
{
  [btn generateBubbleInRandom];
}
 
@end

HWButton:

#import <UIKit/UIKit.h>
 
@interface HWButton : UIButton
 
@property (nonatomic, assign) CGFloat maxLeft;
@property (nonatomic, assign) CGFloat maxRight;
@property (nonatomic, assign) CGFloat maxHeight;
@property (nonatomic, assign) CGFloat duration;
@property (nonatomic, strong) NSArray *images;
 
- (instancetype)initWithFrame:(CGRect)frame maxLeft:(CGFloat)maxLeft maxRight:(CGFloat)maxRight maxHeight:(CGFloat)maxHeight;
 
- (void)generateBubbleWithImage:(UIImage *)image;
 
- (void)generateBubbleInRandom;
 
@end
 
#import "HWButton.h"
 
@implementation HWButton
{
  CGPoint _startPoint;
  CGFloat _maxWidth;
  NSMutableSet *_recyclePool;
  NSMutableArray *_array;
}
 
- (instancetype)initWithFrame:(CGRect)frame maxLeft:(CGFloat)maxLeft maxRight:(CGFloat)maxRight maxHeight:(CGFloat)maxHeight
{
  self = [super initWithFrame:frame];
  if (self) {
    _maxHeight = maxHeight;
    _maxLeft  = maxLeft;
    _maxRight = maxRight;
    
    [self initData];
  }
  return self;
}
 
- (id)initWithCoder:(NSCoder *)aDecoder
{
  self = [super initWithCoder:aDecoder];
  if (self) {
    [self initData];
  }
  return self;
}
 
- (void)initData
{
  _array = @[].mutableCopy;
  _recyclePool = [NSMutableSet set];
}
 
- (void)generateBubbleInRandom
{
  CALayer *layer;
  
  if (_recyclePool.count > 0) {
    layer = [_recyclePool anyObject];
    
    [_recyclePool removeObject:layer];
    
  }else {
    UIImage *image = self.images[arc4random() % self.images.count];
    
    layer = [self createLayerWithImage:image];
  }
  
  [self.layer addSublayer:layer];
  [self generateBubbleWithCAlayer:layer];
}
 
- (void)generateBubbleWithImage:(UIImage *)image
{
  CALayer *layer = [self createLayerWithImage:image];
  
  [self.layer addSublayer:layer];
  [self generateBubbleWithCAlayer:layer];
}
 
- (void)generateBubbleWithCAlayer:(CALayer *)layer
{
  _maxWidth = _maxLeft + _maxRight + self.bounds.size.width;
  
  _startPoint = CGPointMake(self.frame.size.width / 2, 0);
  
  CGPoint endPoint = CGPointMake(_maxWidth * [self randomFloat] - _maxLeft, -_maxHeight);
  CGPoint controlPoint1 = CGPointMake(_maxWidth * [self randomFloat] - _maxLeft, -_maxHeight * 0.2);
  CGPoint controlPoint2 = CGPointMake(_maxWidth * [self randomFloat] - _maxLeft, -_maxHeight * 0.6);
  
  CGMutablePathRef curvedPath = CGPathCreateMutable();
  CGPathMoveToPoint(curvedPath, NULL, _startPoint.x, _startPoint.y);
  CGPathAddCurveToPoint(curvedPath, NULL, controlPoint1.x, controlPoint1.y, controlPoint2.x, controlPoint2.y, endPoint.x, endPoint.y);
  
  CAKeyframeAnimation *keyFrame = [CAKeyframeAnimation animation];
  keyFrame.keyPath = @"position";
  keyFrame.path = CFAutorelease(curvedPath);
  keyFrame.duration = self.duration;
  keyFrame.calculationMode = kCAAnimationPaced;
  
  [layer addAnimation:keyFrame forKey:@"keyframe"];
  
  CABasicAnimation *scale = [CABasicAnimation animation];
  scale.keyPath = @"transform.scale";
  scale.toValue = @1;
  scale.fromValue = [NSValue valueWithCATransform3D:CATransform3DMakeScale(0.1, 0.1, 0.1)];
  scale.duration = 0.5;
  
  CABasicAnimation *alpha = [CABasicAnimation animation];
  alpha.keyPath = @"opacity";
  alpha.fromValue = @1;
  alpha.toValue = @0.1;
  alpha.duration = self.duration * 0.4;
  alpha.beginTime = self.duration - alpha.duration;
  
  CAAnimationGroup *group = [CAAnimationGroup animation];
  group.animations = @[keyFrame, scale, alpha];
  group.duration = self.duration;
  group.delegate = self;
  group.timingFunction = [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionEaseOut];
  group.fillMode = kCAFillModeForwards;
  group.removedOnCompletion = NO;
  [layer addAnimation:group forKey:@"group"];
  
  [_array addObject:layer];
}
 
- (CGFloat)randomFloat
{
  return (arc4random() % 100)/100.0f;
}
 
- (CALayer *)createLayerWithImage:(UIImage *)image
{
  CGFloat scale = [UIScreen mainScreen].scale;
  CALayer *layer = [CALayer layer];
  layer.frame  = CGRectMake(0, 0, image.size.width / scale, image.size.height / scale);
  layer.contents = (__bridge id)image.CGImage;;
  return layer;
}
 
- (void)animationDidStop:(CAAnimation *)anim finished:(BOOL)flag
{
  if (flag) {
    CALayer *layer = [_array firstObject];
    [layer removeAllAnimations];
    [layer removeFromSuperlayer];
    [_array removeObject:layer];
    [_recyclePool addObject:layer];
  }
}
 
@end

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • iOS开发之通过银行卡号获取所属银行名称

    iOS开发之通过银行卡号获取所属银行名称

    本文给大家分享一段代码关于ios通过银行卡号获取所属银行名称,代码简单易懂,在项目开发中经常会遇到这样的功能,需要的朋友一起学习吧
    2016-11-11
  • iOS图片拉伸技巧小结

    iOS图片拉伸技巧小结

    这篇文章主要为大家详细介绍了iOS图片拉伸的技巧,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-01-01
  • objective-c实现点到直线的距离及与垂足的交点

    objective-c实现点到直线的距离及与垂足的交点

    这篇文章主要给大家介绍了利用objective-c实现点到直线的距离及与垂足的交点的相关资料,文中给出了详细的实现思路和实现代码,对大家具有一定的参考价值,需要的朋友们下面来一起看看吧。
    2017-04-04
  • iOS中设置view圆角化的四种方法示例

    iOS中设置view圆角化的四种方法示例

    最近因为工作的原因,遇到view圆角优化的问题,所以将实现的几种方法总结分享出来,下面这篇文章主要给大家介绍了关于iOS中设置view圆角化的四种方法,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴,下面随着小编来一起学习学习吧。
    2017-09-09
  • IOS UIView的生命周期的实例详解

    IOS UIView的生命周期的实例详解

    这篇文章主要介绍了IOS UIView的生命周期的实例详解的相关资料,希望通过本文大家能掌握理解这部分内容,需要的朋友可以参考下
    2017-09-09
  • iOS 隐藏tabbar代码详解

    iOS 隐藏tabbar代码详解

    这篇文章主要介绍了iOS 隐藏tabbar代码详解的相关资料,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2016-07-07
  • iOS开发教程之APP内部切换语言的实现方法

    iOS开发教程之APP内部切换语言的实现方法

    这篇文章主要给大家介绍了关于iOS开发教程之APP内部切换语言的实现方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-02-02
  • iOS利用NSAttributedString实现图文混排效果示例

    iOS利用NSAttributedString实现图文混排效果示例

    iOS7以后,因为TextKit的强大,可以用NSAttributedString很方便的实现图文混排(主要是利用了NSTextAttachment),所以下面这篇文章主要给大家介绍了关于iOS利用NSAttributedString实现图文混排效果的相关资料,需要的朋友可以参考借鉴,下面来一起看看吧。
    2017-10-10
  • iOS UIView常见属性方法小结

    iOS UIView常见属性方法小结

    本文通过实例代码给大家详细介绍了iOS UIView常见属性方法,非常不错,需要的朋友参考下吧
    2016-12-12
  • iOS开发中CALayer使用的基本教程

    iOS开发中CALayer使用的基本教程

    这篇文章主要介绍了iOS开发中CALayer使用的基本教程,代码基于传统的Objective-C,需要的朋友可以参考下
    2015-11-11

最新评论