CALayer之CAEmitterLayer粒子发射器

先发一下CAEmitterLayer做成的demon效果:

snow.gif

like.gif

看过GIF图之后大家应该对CAEmitterLayer充满了好奇,这些该如何实现呢,各位莫慌,只要你耐心看下去,实现这些效果都是小 case。

CAEmitterLayer与CAEmitterCell简介

CAEmitterLayer(粒子发射器)继承自CALayer,是CALayer众多子类中的一个,提供了一个基于Core Animation的粒子发射系统,粒子用CAEmitterCell来初始化,一个单独的CAEmitterLayer可同时支持多个CAEmitterCell。总的来说CAEmitterLayer是用来发射粒子的,是容器,而它所发射的粒子就是CAEmitterCell,CAEmitterCell用来执行具体的动画效果。当然除了图中的两种效果之外,CAEmitterLayer还可以实现如:红包雨、烟花、爆炸、火焰等效果。

CAEmitterLayerCAEmitterCell的属性详解

CAEmitterLayer属性

/** 用于存储cell */
open var emitterCells: [CAEmitterCell]?    
/** 粒子的产生率,默认1;每个粒子cell的产生率乘以这个粒子产生系数,得出每一秒产生这个粒子的个数。 即:每秒粒子产生个数 = layer.birthRate * cell.birthRate */
open var birthRate: Float 
/** 粒子的生命周期,以秒为单位。默认0 */    
open var lifetime: Float 
/** 决定发射源的中心点 */
open var emitterPosition: CGPoint 
/** 发射器在z平面的位置 */
open var emitterZPosition: CGFloat 
/** 决定发射源的大小 */   
open var emitterSize: CGSize
 /** 发射器的深度 */
open var emitterDepth: CGFloat 
/** 表示粒子从什么形状发射出来,它并不是表示粒子自己的形状。是一个枚举类型,提供如下类型可供选择: 
* kCAEmitterLayerPoint 点形状,发射源的形状就是一个点
* kCAEmitterLayerLine 线形状,发射源的形状是一条线
* kCAEmitterLayerRectangle 矩形状,发射源的形状是一个矩形
* kCAEmitterLayerCuboid 立体矩形形状(3D),发射源是一个立体矩形,这里要生效的话需要设置z方向的数据,如果不设置就同矩形状
* kCAEmitterLayerCircle 圆形形状,发射源是一个圆形
* kCAEmitterLayerSphere 立体圆形(3D),三维的圆形,同样需要设置z方向数据,不设置则通二维一样
*/
open var emitterShape: String
 /** 发射模式,这个字段规定了在特定形状上发射的具体形式是什么。它的作用其实就是进一步决定发射的区域是在发射形状的哪一部份,提供如下类型可供选择: 
* kCAEmitterLayerPoints 点模式,发射器是以点的形式发射粒。发射点就是形状的某个特殊的点,比如shap是一个点的话,那么这个点就是中心点,如果是圆形,那么就是圆心
* kCAEmitterLayerOutline 轮廓模式,从形状的边界上发射粒子
* kCAEmitterLayerSurface 表面模式,从形状的表面上发射粒子
* kCAEmitterLayerVolume 是相对于3D形状的“球体内”或“立方体内”发射
*/    
open var emitterMode: String 
/** 渲染模式,,提供如下类型可供选择:
* kCAEmitterLayerUnordered 粒子无序出现
* kCAEmitterLayerOldestFirst 声明久的粒子会被渲染在最上层
* kCAEmitterLayerOldestLast 年轻的粒子会被渲染在最上层
* kCAEmitterLayerBackToFront 粒子的渲染按照Z轴的前后顺序进行
* kCAEmitterLayerAdditive 粒子混合
*/    
open var renderMode: String 
/** 是否开启三维效果 */   
open var preservesDepth: Bool 
/** 粒子速度系数, 默认1.0;粒子速度 = layer.velocity * cell.velocity */    
open var velocity: Float 
/** 粒子的缩放比例系数, 默认1.0,计算方式同上 */   
open var scale: Float 
/** 自旋转速度系数, 默认1.0,计算方式同上 */    
open var spin: Float
/** 随机数发生器 */    
open var seed: UInt32

CAEmitterCell属性

/** 粒子名字, 默认为nil */
open var name: String?
/** 粒子是否渲染 */    
open var isEnabled: Bool
/** 粒子的产生率,默认0 */    
open var birthRate: Float  
/** 粒子的生命周期,以秒为单位。默认0 */
open var lifetime: Float
/** 粒子的生命周期的范围,以秒为单位。默认0 */
open var lifetimeRange: Float
/** 指定纬度,纬度角代表了在x-z轴平面坐标系中与x轴之间的夹角,默认0 */
open var emissionLatitude: CGFloat
/** 指定经度,经度角代表了在x-y轴平面坐标系中与x轴之间的夹角,默认0 */
open var emissionLongitude: CGFloat
/** 发射角度范围,默认0,以锥形分布开的发射角度。角度用弧度制。粒子均匀分布在这个锥形范围内 */    
open var emissionRange: CGFloat
/** 速度和速度范围,两者默认0 */
open var velocity: CGFloat
open var velocityRange: CGFloat
/** x,y,z方向上的加速度分量,三者默认都是0 */    
open var xAcceleration: CGFloat
open var yAcceleration: CGFloat
open var zAcceleration: CGFloat
/** 缩放比例, 默认是1 */    
open var scale: CGFloat
/** 缩放范围与缩放速度, 默认是0 */    
open var scaleRange: CGFloat
open var scaleSpeed: CGFloat
/** 粒子的平均旋转速度和范围,默认是0 */    
open var spin: CGFloat
open var spinRange: CGFloat
/** 粒子的颜色,默认白色 */    
open var color: CGColor?
/** 粒子颜色red、green、blue、alpha能改变的范围,默认0 */    
open var redRange: Float
open var greenRange: Float
open var blueRange: Float
open var alphaRange: Float
/**  粒子颜色red,green,blue,alpha在生命周期内的改变速度,默认都是0 */ 
open var redSpeed: Float
open var greenSpeed: Float
open var blueSpeed: Float
open var alphaSpeed: Float
/** 粒子的内容,为CGImageRef的对象 */    
open var contents: Any?
/** 渲染范围 */    
open var contentsRect: CGRect
/** 内容缩放 */    
open var contentsScale: CGFloat
/** 粒子里面的粒子,可以设置粒子之间的依托关系 */    
open var emitterCells: [CAEmitterCell]?
/** 以下三种是一种叫做拉伸过滤的算法,它作用于原图的像素上并根据需要生成新的像素显示在屏幕上。具有以下三种模式:
* kCAFilterLinear 这个过滤器采用双线性滤波算法,它在大多数情况下都表现良好。双线性滤波算法通过对多个像素取样最终生成新的值,得到一个平滑的表现不错的拉伸。但是当放大倍数比较大的时候图片就模糊不清了。
* kCAFilterTrilinear 三线性滤波算法,存储了多个大小情况下的图片(也叫多重贴图),并三维取样,同时结合大图和小图的存储进而得到最后的结果。
* kCAFilterNearest 是一种比较武断的方法。从名字不难看出,这个算法(也叫最近过滤)就是取样最近的单像素点而不管其他的颜色。这样做非常快,也不会使图片模糊。但是,最明显的效果就是,会使得压缩图片更糟,图片放大之后也显得块状或是马赛克严重。
*/
/** 缩小图片,是默认的过滤器 */    
open var minificationFilter: String
/** 放大图片 */ 
open var magnificationFilter: String
/** 减小大小的因子 */ 
open var minificationFilterBias: Float

对于滤波器有兴趣的童鞋可以参考拉伸过滤这篇文章。

CAEmitterCell中,所有带有range的参数是一种范围内随即发生的事件。几个🌰:

snowCell.emissionLongitude = CGFloat(M_PI_2)
snowCell.emissionRange = CGFloat(M_PI_4)

具体效果如下:


屏幕快照 2018-09-29 下午3.26.10.png

实战演练:

demon1❄️:

class SnowViewController: UIViewController {
    private var imageArray:Array<UIImage> = []
    private var timeArray:Array<NSNumber> = []
    private var totalTime:TimeInterval =  0
    
    override func viewDidLoad() {
        super.viewDidLoad()
        self.title = "雪"
        self.view.backgroundColor = UIColor.white
        self.view.addSubview(imageView!)
        imageView?.layer.addSublayer(snowLayer!)
    }
    /** snowLayer懒加载 */
    lazy var snowLayer:CAEmitterLayer? = {
        var snowLayer = CAEmitterLayer()
        /** 发射源的形状 是枚举类型 */
        snowLayer.emitterShape = kCAEmitterLayerLine
        /** 发射模式 枚举类型 */
        snowLayer.emitterMode = kCAEmitterLayerSurface
        /** 发射源的size 决定了发射源的大小 */
        snowLayer.emitterSize = self.view.frame.size
        /** 发射源的位置 */
        snowLayer.emitterPosition = CGPoint(x: self.view.frame.size.width*0.5, y: -10)
        /** 每秒产生的粒子数量的系数 */
        snowLayer.birthRate = 6.0
        
        var snowCell = CAEmitterCell()
        /** 粒子的内容 是CGImageRef类型的 */
        snowCell.contents = UIImage(named: "snow_white")?.cgImage
        /** 每秒产生的粒子数量 */
        snowCell.birthRate = 10
        /** 粒子的生命周期 */
        snowCell.lifetime = 200.0
        /** 粒子的速度 */
        snowCell.velocity = 4.0
        
        /** 粒子再y方向的加速的 */
        snowCell.yAcceleration = 20.0
        snowCell.velocityRange = 10.0
        /** 粒子的缩放比例 */
        snowCell.scale = 0.15
        snowCell.scaleRange = 0.14
        /** 向下 */
        snowCell.emissionLongitude = CGFloat(M_PI_2);
        snowCell.emissionRange = CGFloat(M_PI_4);
        /** 粒子添加到CAEmitterLayer上 */
        snowLayer.emitterCells = [snowCell]
        return snowLayer
    }()
    
    lazy var imageView:UIImageView? = {
        /** 加载GIF图片,并转化为data类型 */
        let path = Bundle.main.path(forResource: "snow.gif", ofType: nil)
        let data = NSData(contentsOfFile: path!)
        /** 从data中读取数据,转换为CGImageSource */
        let imageSource = CGImageSourceCreateWithData(data!, nil)
        let imageCount = CGImageSourceGetCount(imageSource!)
        for i in 0..<imageCount {
            /** 取出图片 */
            let cgImage = CGImageSourceCreateImageAtIndex(imageSource!, i, nil)
            let image = UIImage(cgImage: cgImage!)
            imageArray.append(image)
            /** 取出持续时间 */
            let properties = CGImageSourceCopyPropertiesAtIndex(imageSource!, i, nil)! as NSDictionary
            let gifDict = properties.value(forKey: kCGImagePropertyGIFDictionary as String)  as? NSDictionary
            let frameDuration = gifDict![kCGImagePropertyGIFDelayTime] as? NSNumber
            totalTime += (frameDuration?.doubleValue)!
        }
        let imageView = UIImageView(frame: self.view.bounds)
        /** 设置imageview的属性 */
        imageView.animationImages = imageArray
        imageView.animationDuration = totalTime
        imageView.animationRepeatCount = 100
        /** 开始动画 */
        imageView.startAnimating()
        return imageView
    }()
}

demon2👍:

class LikeButton:UIButton{
    var explosionLayer:CAEmitterLayer?
    override func awakeFromNib() {
        super.awakeFromNib()
        setupExplosion()
    }
    
    override init(frame: CGRect) {
        super.init(frame: frame)
        setupExplosion()
    }
    
    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
    
    func setupExplosion() {
        let explosionCell = CAEmitterCell()
        explosionCell.name = "explosionCell"
        /** 粒子的内容 是CGImageRef类型的 */
        explosionCell.contents = UIImage(named: "spark_blue")?.cgImage
        explosionCell.alphaSpeed = -1.0
        explosionCell.alphaRange = 0.10
        /** 粒子的生命周期 */
        explosionCell.lifetime = 1.0
        explosionCell.lifetimeRange = 0.1
        /** 粒子的速度 */
        explosionCell.velocity = 40.0
        explosionCell.velocityRange = 10.0
        /** 粒子的缩放比例 */
        explosionCell.scale = 0.08
        explosionCell.scaleRange = 0.02
        explosionCell.birthRate = 500
        
        explosionLayer = CAEmitterLayer()
        explosionLayer?.name = ""
        /** 发射源的形状 是枚举类型 */
        explosionLayer?.emitterShape = kCAEmitterLayerCircle
        /** 发射模式 枚举类型 */
        explosionLayer?.emitterMode = kCAEmitterLayerOutline
        explosionLayer?.renderMode = kCAEmitterLayerOldestFirst
        /** 发射源的size 决定了发射源的大小 */
        explosionLayer?.emitterSize = CGSize(width: self.bounds.size.width+40, height: self.bounds.size.height+40)
        /** 粒子添加到CAEmitterLayer上 */
        explosionLayer?.emitterCells = [explosionCell]
        explosionLayer?.birthRate = 0
        self.layer.addSublayer(explosionLayer!)
        explosionLayer?.position = CGPoint(x: self.bounds.size.width*0.5, y: self.bounds.size.height*0.5)
    }
    
    override func layoutSubviews() {
        
        super.layoutSubviews()
    }
    
    override func sendAction(_ action: Selector, to target: Any?, for event: UIEvent?) {
        let animaton = CAKeyframeAnimation(keyPath: "transform.scale")
        self.isSelected = !self.isSelected
        if self.isSelected{
            animaton.values = [1.5,2.0, 0.8, 1.0]
            animaton.duration = 0.5
            animaton.calculationMode = kCAAnimationCubic
            self.layer.add(animaton, forKey: nil)
            self.perform(#selector(startAnimation), with: nil, afterDelay: 0.25)
        }else{
            animaton.values = [0.8, 1.0]
            animaton.duration = 0.4
            self.layer.add(animaton, forKey: nil)
            stopAnimation()
        }
    }
    
   @objc func startAnimation() {
    explosionLayer?.birthRate = 10
    explosionLayer?.beginTime = CACurrentMediaTime()
        self.perform(#selector(stopAnimation), with: nil, afterDelay: 0.25)
    }
    
    @objc func stopAnimation() {
        explosionLayer?.setValue(0, forKey: "birthRate")
        explosionLayer?.removeAllAnimations()
    }
}


class LikeViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
        self.title = "点赞"
        self.view.backgroundColor = UIColor.white
        let likeBtn = LikeButton(frame: CGRect(x: 100, y: 150, width: 30, height: 30))
        likeBtn.setImage(UIImage(named: "dislike"), for: .normal)
        likeBtn.setImage(UIImage(named: "liek_orange"), for: .selected)
        likeBtn.addTarget(self, action: #selector(likeBtnAction(btn:)), for: .touchUpInside)
        self.view.addSubview(likeBtn)
    }
}

总结:这篇文章参考了众多大佬的博客,在此表示非常感谢。如果在阅读的时候,发现文章中存在问题,请大家不吝赐教。

参考资料:https://www.jianshu.com/p/c54ffd7412e7

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容