动画黄金搭档:CADisplayLink CAShapeLayer

2023-10-15 05:20

本文主要是介绍动画黄金搭档:CADisplayLink CAShapeLayer,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

我们在开发中有时会遇到一些看似非常复杂的动画,不知该如何下手,今天的这篇文章中我会讲到如何利用CADisplayLink和CAShapeLayer来构建一些复杂的动画,希望能在你下次构建动画中,给你一些启发。(备注:收藏下来以供学习,如需转载请备注原创:夏树正茂 投稿)

在接下来的文章中,我们会构建如下的一个动画:

8f7a6fe0gw1f6lb4ym7ovg207002z3ym.gif

该动画是在du的轮廓中进行,类似一个镂空效果,轮廓的填充是用双波浪的形式,类似于水流慢慢注入容器的过程。
动画使用CADisplayLink来进行刷新,保证了动画的流程性,利用CAShapeLayer来构建波浪的轮廓,最后利用CALayer的mask属性来实现逐渐填充的过程。

背景知识介绍

在动画创建过程的讲解之前,先介绍一下会使用到的一些知识点:

  • CADisplayLink

  • UIBezierPath

  • CAShapeLayer

  • mask

如果你已经对这些概念有了充分的了解,可以略过背景知识介绍这一节。

1、CADisplayLink

用绘制的方式构建的动画,必然需要不断的刷新绘制的内容来呈现流畅的动画,CADisplayLink就像是一个定时器,每隔几毫秒刷新一次屏幕。能让我们以和屏幕刷新频率相同的频率去刷新我们绘制到屏幕上的内容。
CADisplayLink的使用方式如下:

1
2
3
4
   _displayLink = [CADisplayLink displayLinkWithTarget:self
                                             selector:@selector(updateContent:)];
    [_displayLink addToRunLoop:[NSRunLoop currentRunLoop]
                       forMode:NSRunLoopCommonModes];

当CADisplayLink注册到runloop以后,屏幕刷新的时候就会调用绑定到它上面的target所拥有的selector方法。停止CADisplayLink的运行非常的简单,只需要调用它的invalidate方法。

NSTimer和CADisplayLink有什么不同?

iOS设备的屏幕每秒会刷新60次,正常情况下CADisplayLink在屏幕每次刷新时都会调用,精确度非常高,并且CADisplayLink的使用场合相对专一,适合做UI的不停重绘,比如动画的连续绘制。

NSTimer的使用范围要广泛很多,可以做单次或者循环处理某个任务,精度相比CADisplayLink要低。

2、UIBezierPath

使用UIBezierPath类可以创建基于矢量的路径,它是Core Graphics框架关于CGPathRef类型数据的封装,利用它创建直线或者曲线来构建我们想要的形状,每一个直线段或者曲线段的结束位置就是下一个线段开始的地方。这些连接的直线或者曲线的集合成为subpath。一个UIBezierPath对象的完整路径包括一个或者多个subpath。

创建一个完整的UIBezierPath对象的完整步骤如下:

  • 创建一个Bezier Path对象。

  • 使用方法moveToPoint:去设置初始线段的起点。

  • 添加line或者curve去定义一个或者多个subpath。

  • 修改UIBezierPath对象跟绘图相关的属性。

3、CAShapeLayer

CAShapeLayer是一个通过矢量图形而不是bitmap来绘制的图层子类。CAShapeLayer可以用来绘制所有通过CGPath来表示的形状,上面讲到了可以用UIBezierPath来创建任何你想要的路径,使用CAShapeLayer的属性path配合UIBezierPath创建的路径,就可以呈现出我们想要的形状。
这个形状不一定要闭合,图层路径也不一定是连续不断的,你可以在CAShapeLayer的图层上绘制好几个不同的形状,但是你只有一次机会去设置它的path、lineWith、lineCap等属性,如果你想同时设置几个不同颜色的多个形状,你就需要为每个形状准备一个图层。

下面的示例代码是通过UIBezierPath和CAShapeLayer来创建一个简单的火柴人。

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
   - (void)viewDidLoad {
     [ super  viewDidLoad];
     
     UIBezierPath *path = [[UIBezierPath alloc] init];
     [path moveToPoint:CGPointMake(175, 100)];
     
     [path addArcWithCenter:CGPointMake(150, 100) radius:25 startAngle:0 endAngle:2*M_PI clockwise:YES];
     [path moveToPoint:CGPointMake(150, 125)];
     [path addLineToPoint:CGPointMake(150, 175)];
     [path addLineToPoint:CGPointMake(125, 225)];
     [path moveToPoint:CGPointMake(150, 175)];
     [path addLineToPoint:CGPointMake(175, 225)];
     [path moveToPoint:CGPointMake(100, 150)];
     [path addLineToPoint:CGPointMake(200, 150)];
     
     //create shape layer
     CAShapeLayer *shapeLayer = [CAShapeLayer layer];
     shapeLayer.strokeColor = [UIColor colorWithRed:147/255.0 green:231/255.0 blue:182/255.0 alpha:1].CGColor;
     shapeLayer.fillColor = [UIColor clearColor].CGColor;
     shapeLayer.lineWidth = 5;
     shapeLayer.lineJoin = kCALineJoinRound;
     shapeLayer.lineCap = kCALineCapRound;
     shapeLayer.path = path.CGPath;
     //add it to our view
     [self.view.layer addSublayer:shapeLayer];
}

显示的形状如下:

69.jpg

4、mask

CALayer有一个属性叫做mask,通常被称为蒙版图层,这个属性本身也是CALayer类型,有和其他图层一样的绘制和布局属性。它类似于一个子视图,相对于父图层(即拥有该属性的图层)布局,但是它却不是一个普通的子视图。不同于一般的subLayer,mask定义了父图层的可见区域,简单点说就是最终父视图显示的形态是父视图自身和它的属性mask的交集部分。

8f7a6fe0gw1f6lb50rgqbj20go06mwet.jpg

mask图层的color属性是无关紧要的,真正重要的是它的轮廓,mask属性就像一个切割机,父视图被mask切割,相交的部分会留下,其他的部分则被丢弃。
CALayer的蒙版图层真正厉害的地方在于蒙版图层不局限于静态图,任何有图层构成的都可以作为mask属性,这意味着蒙版可以通过代码甚至是动画实时生成。这也为我们实现示例中波浪的变化提供了支持。

绘制波浪轮廓

我们利用UIBezierPath来绘制波浪的轮廓,通过正弦函数和余弦函数来创建顶部的波浪曲线,在单位为1的右手直角坐标系中的曲线变化如下:

8f7a6fe0gw1f6lb51vscuj20v007sq4f.jpg

可以看到在(-2π , 2π )的范围类,y值在[-1, 1]之间变化。
以正弦曲线为例,它可以表示为y=Asin(ωx+φ)+k,公式中各符号表示的含义:

  • A–振幅,即波峰的高度。

  • (ωx+φ)–相位,反应了变量y所处的位置。

  • φ–初相,x=0时的相位,反映在坐标系上则为图像的左右移动。

  • k–偏距,反映在坐标系上则为图像的上移或下移。

  • ω–角速度,控制正弦周期(单位角度内震动的次数)。

通过上面的函数,我们就能计算出波浪曲线上任意位置的坐标点。通过UIBezierPath的函数addLineToPoint来把这些点连接起来,就构建了波浪形状的path。只要我们的设定相邻两点的间距够小,就能构建出平滑的正弦曲线,构建正弦波浪的代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
   - (UIBezierPath *)createSinPath
{
     UIBezierPath *wavePath = [UIBezierPath bezierPath];
     CGFloat endX = 0;
     for  (CGFloat x = 0; x < self.waveWidth + 1; x += 1) {
         endX=x;
         CGFloat y = self.maxAmplitude * sinf(360.0 / _waveWidth * (x  * M_PI / 180) * self.frequency + self.phase * M_PI/ 180) + self.maxAmplitude;
         if  (x == 0) {
             [wavePath moveToPoint:CGPointMake(x, y)];
         else  {
             [wavePath addLineToPoint:CGPointMake(x, y)];
         }
     }
    
     CGFloat endY = CGRectGetHeight(self.bounds) + 10;
     [wavePath addLineToPoint:CGPointMake(endX, endY)];
     [wavePath addLineToPoint:CGPointMake(0, endY)];
     
     return  wavePath;
}

显示的效果如下:

330.jpg

在这里我们设定了两个正弦曲线上的点的横坐标间距是1,现在来解释一下通过横坐标x来得出y的计算过程:

1
y = self.maxAmplitude * sinf(360.0 / _waveWidth * (x  * M_PI / 180) * self.frequency + self.phase * M_PI/ 180) + self.maxAmplitude;

第一个self.maxAmplitude表示曲线的波峰值,360.0 / _waveWidth计算出单位间距1pixel代表的度数,x * M_PI / 180表示将横坐标值转换为角度。self.frequency表示角速度,即单位面积内波动次数,波浪的大小。self.phase * M_PI/ 180代表上面公式中的初相,通过规律的变化初相,可以制造出曲线上的点动起来的效果,self.maxAmplitude代表偏距,由于我们需要让波浪曲线的波峰在layer的范围内显示,所以需要将整个曲线向下移动波峰大小的单位,因为CALayer使用左手坐标系,所以向下移动需要加上波峰的大小。

让波浪曲线动起来

正弦或者余弦曲线上的点,不论角度如何,在y轴上的变化范围在它的波峰与波谷之间。拿单位正交直角坐标系来说,只要我们规律性的增加角度值,那么曲线上的点就会在[1, -1]之间变化,我们以曲线上x=0的点为例,角度的不断增加,会让它的y值规律性的来回变化:

39.gif

如若曲线上的点都能这样规律的变化,我们就能让波浪曲线浪起来。
要让曲线上所有的点都动起来,在这里我们使用CADisplayLink来不断刷新由UIBezierPath创建的形状,两次刷新之间曲线的变化通过增加初相来实现,代码如下:

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
- (void)startLoading
{
     [_displayLink invalidate];
     self.displayLink = [CADisplayLink displayLinkWithTarget:self
                                                    selector:@selector(updateWave:)];
     [_displayLink addToRunLoop:[NSRunLoop currentRunLoop]
                        forMode:NSRunLoopCommonModes];
}
- (void)updateWave:(CADisplayLink *)displayLink
{
     self.phase += 8; //逐渐累加初相
     self.waveSinLayer.path = [self createSinPath].CGPath;
}
  
- (UIBezierPath *)createSinPath
{
     UIBezierPath *wavePath = [UIBezierPath bezierPath];
     CGFloat endX = 0;
     for  (CGFloat x = 0; x < self.waveWidth + 1; x += 1) {
         endX=x;
         CGFloat y = self.maxAmplitude * sinf(360.0 / _waveWidth * (x  * M_PI / 180) * self.frequency + self.phase * M_PI/ 180) + self.maxAmplitude;
         if  (x == 0) {
             [wavePath moveToPoint:CGPointMake(x, y)];
         else  {
             [wavePath addLineToPoint:CGPointMake(x, y)];
         }
     }
     
     CGFloat endY = CGRectGetHeight(self.bounds) + 10;
     [wavePath addLineToPoint:CGPointMake(endX, endY)];
     [wavePath addLineToPoint:CGPointMake(0, endY)];
     
     return  wavePath;
}

把CAShapeLayer的背景色设置为淡红色,波浪曲线会在Layer的bounds类波动,动起来的波浪曲线如下:

36.gif

构建波浪上升的镂空效果

到目前为止,我们利用CAShapeLayer、UIBezierPath以及CADisplayLink实现了动起来的波浪效果,下面我们需要实现的是在du的轮廓里,水波不断上升填充的过程,整个动画过程中,会呈现一个du的镂空效果,在它轮廓之外的水波则不会显示,这样的效果需要借助CALayer的mask属性来实现。

我们需要的动画素材如下:

8f7a6fe0gw1f6lb55xrpgj20bm04iaa5.jpg

将这三个图片位置设置为一样,底层放置动画中一直显示的底层轮廓,中间层用以实现余弦波浪,最外层用于实现正弦波浪,将中间层和最外层图片的mask属性赋值为我们创建的用来呈现余弦波浪和正弦波浪的CAShapeLayer,这样动画开始后,利用CADisplayLink来不断刷新两个CAShapeLayer的path来让波浪浪起来,再利用CABasicAnimation来对两个CAShapeLayer的position进行动画,实现从下到上的填充效果。我们想要的效果就完成了:

8f7a6fe0gw1f6lb4ym7ovg207002z3ym.gif

完整的代码示例在这里

参考

  • UIBezierPath

  • CAShapeLayer

  • 图层蒙版

转载于:https://www.cnblogs.com/pioneerMax/p/6136846.html

这篇关于动画黄金搭档:CADisplayLink CAShapeLayer的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/215576

相关文章

最好用的WPF加载动画功能

《最好用的WPF加载动画功能》当开发应用程序时,提供良好的用户体验(UX)是至关重要的,加载动画作为一种有效的沟通工具,它不仅能告知用户系统正在工作,还能够通过视觉上的吸引力来增强整体用户体验,本文给... 目录前言需求分析高级用法综合案例总结最后前言当开发应用程序时,提供良好的用户体验(UX)是至关重要

基于Python实现PDF动画翻页效果的阅读器

《基于Python实现PDF动画翻页效果的阅读器》在这篇博客中,我们将深入分析一个基于wxPython实现的PDF阅读器程序,该程序支持加载PDF文件并显示页面内容,同时支持页面切换动画效果,文中有详... 目录全部代码代码结构初始化 UI 界面加载 PDF 文件显示 PDF 页面页面切换动画运行效果总结主

Qt QWidget实现图片旋转动画

《QtQWidget实现图片旋转动画》这篇文章主要为大家详细介绍了如何使用了Qt和QWidget实现图片旋转动画效果,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 一、效果展示二、源码分享本例程通过QGraphicsView实现svg格式图片旋转。.hpjavascript

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

Flutter 进阶:绘制加载动画

绘制加载动画:由小圆组成的大圆 1. 定义 LoadingScreen 类2. 实现 _LoadingScreenState 类3. 定义 LoadingPainter 类4. 总结 实现加载动画 我们需要定义两个类:LoadingScreen 和 LoadingPainter。LoadingScreen 负责控制动画的状态,而 LoadingPainter 则负责绘制动画。

用Unity2D制作一个人物,实现移动、跳起、人物静止和动起来时的动画:中(人物移动、跳起、静止动作)

上回我们学到创建一个地形和一个人物,今天我们实现一下人物实现移动和跳起,依次点击,我们准备创建一个C#文件 创建好我们点击进去,就会跳转到我们的Vision Studio,然后输入这些代码 using UnityEngine;public class Move : MonoBehaviour // 定义一个名为Move的类,继承自MonoBehaviour{private Rigidbo

动画AnimationDrawable、转动

现实开发中:很多地方都用到 点击动画的特效; 本案例本人做了三个关于“动” 画 的效果; 先上图: 总体图: A: B: 1:点击图片按钮,效果是:图片闪动; 通过在xml中定义:标签:animation-list来实现点击动画的效果;  是否循环标签:oneshot ;   时间间隔标签:duration ; 要显示的图片标签:drawable ;

13 transition数组的动画使用

划重点 动画:transitiontransition-group :数组动画数组的 添加 / 删除 豆腐粉丝汤 清淡又健康 <!DOCTYPE html><html lang="en"><head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><me

12 动画transition的使用2

划重点 Vue 动画:transition / transform在动画周期中执行动动画(上一篇是通过动画样式控制动画) 清蒸扇贝 <!DOCTYPE html><html lang="en"><head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><

【前端】animation动画以及利用vue制作简单的透明度改变动画,包含vue生命周期实现

一. 问题描述 想做一个文字透明度从1到0然后再从0到1的css动画。 二. 代码写法 2.1 animation写法 2.1.1 animation属性key 2.1.2 代码展示 <!DOCTYPE html><html lang="en"><head><meta charset="UTF-8"><meta name="viewport" content="width=de