#import "DDJelloView.h"

#define SYS_DEVICE_WIDTH    ([[UIScreen mainScreen] bounds].size.width)                  // 屏幕宽度

#define SYS_DEVICE_HEIGHT   ([[UIScreen mainScreen] bounds].size.height)                 // 屏幕长度

#define MIN_HEIGHT          50                                                          // 图形最小高度

@interface DDJelloView ()

@property (nonatomic, assign) CGFloat mHeight;

@property (nonatomic, assign) CGFloat curveX;               // r5点x坐标

@property (nonatomic, assign) CGFloat curveY;               // r5点y坐标

@property (nonatomic, strong) UIView *curveView;            // r5红点

@property (nonatomic, strong) CAShapeLayer *shapeLayer;

@property (nonatomic, strong) CADisplayLink *displayLink;

@property (nonatomic, assign) BOOL isAnimating;

@end

@implementation DDJelloView

static NSString *kX = @"curveX";

static NSString *kY = @"curveY";

- (instancetype)initWithFrame:(CGRect)frame

{

self = [super initWithFrame:frame];

if(self)

{

[self addObserver:self forKeyPath:kX options:NSKeyValueObservingOptionNew context:nil];

[self addObserver:self forKeyPath:kY options:NSKeyValueObservingOptionNew context:nil];

[self configShapeLayer];

[self configCurveView];

[self configAction];

}

return self;

}

- (void)dealloc {

[self removeObserver:self forKeyPath:kX];

[self removeObserver:self forKeyPath:kY];

}

- (void)drawRect:(CGRect)rect

{

}

-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context {

if ([keyPath isEqualToString:kX] || [keyPath isEqualToString:kY]) {

[self updateShapeLayerPath];

}

}

#pragma mark -

#pragma mark - Configuration

- (void)configAction

{

_mHeight = 100;                       // 手势移动时相对高度

_isAnimating = NO;                    // 是否处于动效状态

// 手势

UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(handlePanAction:)];

self.userInteractionEnabled = YES;

[self addGestureRecognizer:pan];

// CADisplayLink默认每秒运行60次calculatePath是算出在运行期间_curveView的坐标,从而确定_shapeLayer的形状

_displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(calculatePath)];

[_displayLink addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];

_displayLink.paused = YES;

}

- (void)configShapeLayer

{

_shapeLayer = [CAShapeLayer layer];

//    _shapeLayer.fillColor = [UIColor colorWithRed:57/255.0 green:67/255.0 blue:89/255.0 alpha:1.0].CGColor;

_shapeLayer.fillColor = [UIColor redColor].CGColor;

[self.layer addSublayer:_shapeLayer];

}

- (void)configCurveView

{

// _curveView就是r5点

self.curveX = SYS_DEVICE_WIDTH/2.0;       // r5点x坐标

self.curveY = MIN_HEIGHT;                 // r5点y坐标

_curveView = [[UIView alloc] initWithFrame:CGRectMake(_curveX, _curveY, 3, 3)];

_curveView.backgroundColor = [UIColor redColor];

[self addSubview:_curveView];

}

#pragma mark -

#pragma mark - Action

- (void)handlePanAction:(UIPanGestureRecognizer *)pan

{

if(!_isAnimating)

{

if(pan.state == UIGestureRecognizerStateChanged)

{

// 手势移动时,_shapeLayer跟着手势向下扩大区域

CGPoint point = [pan translationInView:self];

// 这部分代码使r5红点跟着手势走

_mHeight = point.y*0.7 + MIN_HEIGHT;

self.curveX = SYS_DEVICE_WIDTH/2.0 + point.x;

self.curveY = _mHeight > MIN_HEIGHT ? _mHeight : MIN_HEIGHT;

_curveView.frame = CGRectMake(_curveX,

_curveY,

_curveView.frame.size.width,

_curveView.frame.size.height);

}

else if (pan.state == UIGestureRecognizerStateCancelled ||

pan.state == UIGestureRecognizerStateEnded ||

pan.state == UIGestureRecognizerStateFailed)

{

// 手势结束时,_shapeLayer返回原状并产生弹簧动效

_isAnimating = YES;

_displayLink.paused = NO;           //开启displaylink,会执行方法calculatePath.

// 弹簧动效

[UIView animateWithDuration:1.0

delay:0.0

usingSpringWithDamping:0.5

initialSpringVelocity:0

options:UIViewAnimationOptionCurveEaseInOut

animations:^{

// 曲线点(r5点)是一个view.所以在block中有弹簧效果.然后根据他的动效路径,在calculatePath中计算弹性图形的形状

_curveView.frame = CGRectMake(SYS_DEVICE_WIDTH/2.0, MIN_HEIGHT, 3, 3);

} completion:^(BOOL finished) {

if(finished)

{

_displayLink.paused = YES;

_isAnimating = NO;

}

}];

}

}

}

- (void)updateShapeLayerPath

{

// 更新_shapeLayer形状

UIBezierPath *tPath = [UIBezierPath bezierPath];

[tPath moveToPoint:CGPointMake(0, 0)];                              // r1点

[tPath addLineToPoint:CGPointMake(SYS_DEVICE_WIDTH, 0)];            // r2点

[tPath addLineToPoint:CGPointMake(SYS_DEVICE_WIDTH,  MIN_HEIGHT)];  // r4点

[tPath addQuadCurveToPoint:CGPointMake(0, MIN_HEIGHT)

controlPoint:CGPointMake(_curveX, _curveY)]; // r3,r4,r5确定的一个弧线

[tPath closePath];

_shapeLayer.path = tPath.CGPath;

}

- (void)calculatePath

{

// 由于手势结束时,r5执行了一个UIView的弹簧动画,把这个过程的坐标记录下来,并相应的画出_shapeLayer形状

CALayer *layer = _curveView.layer.presentationLayer;

self.curveX = layer.position.x;

self.curveY = layer.position.y;

}

@end

iOS开发 弹簧效果的更多相关文章

  1. ios开发抽屉效果的封装使用

    #import "DragerViewController.h" #define screenW [UIScreen mainScreen].bounds.size.width @ ...

  2. iOS开发——UI篇&提示效果

    提示效果 关于iOS开发提示效果是一个很常见的技术,比如我们平时点击一个按钮,实现回馈,或者发送网络请求的时候! 技术点: 一:View UIAlertView UIActionSheet 二:控制器 ...

  3. iOS开发之各种动画各种页面切面效果

    因工作原因,有段时间没发表博客了,今天就发表篇博客给大家带来一些干货,切勿错过哦.今天所介绍的主题是关于动画的,在之前的博客中也有用到动画的地方,今天就好好的总结一下iOS开发中常用的动画.说道动画其 ...

  4. iOS开发 QQ粘性动画效果

    QQ(iOS)客户端的粘性动画效果 时间 2016-02-17 16:50:00  博客园精华区 原文  http://www.cnblogs.com/ziyi--caolu/p/5195615.ht ...

  5. iOS开发笔记7:Text、UI交互细节、两个动画效果等

    Text主要总结UILabel.UITextField.UITextView.UIMenuController以及UIWebView/WKWebView相关的一些问题. UI细节主要总结界面交互开发中 ...

  6. ios开发中超简单抽屉效果(MMDrawerController)的实现

    ios开发中,展示类应用通常要用到抽屉效果,由于项目需要,本人找到一个demo,缩减掉一些不常用的功能,整理出一个较短的实例. 首先需要给工程添加第三方类库 MMDrawerController: 这 ...

  7. 【转】iOS开发之各种动画各种页面切面效果

    原文: http://www.cnblogs.com/ludashi/p/4160208.html?utm_source=tuicool 因工作原因,有段时间没发表博客了,今天就发表篇博客给大家带来一 ...

  8. iOS开发之虾米音乐频道选择切换效果分析与实现

    今天博客的内容比较简单,就是看一下虾米音乐首页中频道选择的一个动画效果的实现.之前用mask写过另外一种Tab切换的一种效果,网易云音乐里边的一种Tab切换效果,详情请移步于"视错觉:从一个 ...

  9. iOS开发探索-高斯模糊&毛玻璃效果

    iOS开发中有的时候需要将图片设置模糊,来实现特定的效果获取更好的用户体验, iOS7之后半透明模糊效果得到大范围使用的比较大,现在也可以看到很多应用局部用到了图片模糊效果,可以通过高斯模糊和毛玻璃效 ...

随机推荐

  1. js鲸鱼

    <!DOCTYPE html> <html lang="en" xmlns="http://www.w3.org/1999/xhtml"> ...

  2. C#编程利器之三:接口(Interface)【转】

    C#编程利器之三:接口(Interface) C#接口是一个让很多初学者容易迷糊的东西,用起来好象很简单,定义接口,然后在里面定义方法,通过继承与他的子类来完成具体的实现.但没有真正认识接口的作用的时 ...

  3. oracle从零开始学习笔记

    查询现有数据库:select name from V$database; 解锁用户scott:alter user scott account unlock; 普通用户连接:conn scott 默认 ...

  4. Centos7安装配置NFS服务和挂载

    现在有3台服务器 s1(主),s2(从), s3(从)需要实现文件实时同步,我们可以安装Nfs服务端和客户端来实现! 一.安装 NFS 服务器所需的软件包:   1 yum install -y nf ...

  5. 部署移动BI必须要考虑的五大因素

    随着大屏智能手机和平板电脑的普及,商业智能分析应用程序也需要移动化.由于用户并不会长时间在办公桌前工作.在新的设备以及新的用户的共同促进下,移动BI即将成为主流. 但是,所有人都应该清楚这样一个事实: ...

  6. magento 备份

    magento 备份分为“文件备份”和“数据备份” 我们先来讲下“数据备份” 数据备份的方法有 通过数据库软件直接导出magento使用的数据库,使用mysql命名或者phpmyadmin,导出来就好 ...

  7. JAVA的网络编程

    网络编程 网络编程对于很多的初学者来说,都是很向往的一种编程技能,但是很多的初学者却因为很长一段时间无法进入网络编程的大门而放弃了对于该部分技术的学习. 在 学习网络编程以前,很多初学者可能觉得网络编 ...

  8. html5中的beginPath与stroke

    名词解释: 定义和用法 beginPath() 方法在一个画布中开始子路径的一个新的集合. 语法 beginPath() 描述 beginPath() 丢弃任何当前定义的路径并且开始一条新的路径.它把 ...

  9. ajax异步处理时,如何在JS中获取从Servlet或者Action中session,request

    ssh项目中,我需要登陆某个页面(如a.jsp),通过onblur()鼠标失去焦点后来触发js函数(函数是ajax请求)请求到相应的action,处理完成后将数据存放到session对象里面,然后在a ...

  10. URL中文乱码处理总结(转)

    转自:http://www.cnblogs.com/xirongliu/archive/2012/09/06/2674196.html 问题:传递中文参数的时候,接收页面出现乱码问题?当跨域操作,两套 ...