iOS 功能拓展按钮

最终实现的效果为:

checkbox.gif
因为最近做项目的时候要做一个拓展选项功能,当时想弄成
`tabbar`形式,中间弄一个拓展功能,类似于微博那种,但是要弄
成这样就得把结构修改了,然后就想自定义一个了。
进入正题

动画的开始看到+按钮,点击的时候周围会出现两个圆一闪而逝,然后内部的+开始旋转,vertical方向的线条多旋转了M_PI_2,因为是各自动画,所以horizontalvertical分别有自己的shapeLayer

这里有个值得提的就是,必须自身有frame,才可以设置锚点,有时候可能会这样做:

let shapeLayer = CAShapeLayer()
let path = UIBezierPath()
//path 绘制代码
shapeLayer.path = path.cgPath

这样效果是可以出来,但是可以去看一下,Layer.frame是没有的,position也就没有,旋转轴就没办法控制了。

所以这里我是设置layer.frame,然后path基于layer去绘制。

//垂直直线
self.verticalLayer = CAShapeLayer.init()
        
//设置frame才能设置旋转点
self.verticalLayer.frame = CGRect.init(x: self.frame.size.width / 2 - 2, y: 10, width: 4, height: self.frame.size.width - 20)
        
let verticalPath = UIBezierPath.init()
        
verticalPath.move(to: CGPoint.init(x: 2,y: 0))
        
verticalPath.addLine(to: CGPoint.init(x: 2, y: self.frame.size.width - 20))
        
self.verticalLayer.path = verticalPath.cgPath
        
self.verticalLayer.strokeColor = UIColor.white.cgColor
        
self.verticalLayer.cornerRadius = 2
        
self.verticalLayer.lineWidth = 4
        
self.verticalLayer.masksToBounds = true
        
self.layer.addSublayer(self.verticalLayer)
        
//水平直线
self.horizontalLayer = CAShapeLayer()
        
horizontalLayer.frame = CGRect.init(x: 10, y: self.frame.size.width / 2 - 2, width: self.frame.size.width - 20, height: 4)
        
let horizontalPath = UIBezierPath.init()
        
horizontalPath.move(to: CGPoint.init(x: 0, y: 2))
        
horizontalPath.addLine(to: CGPoint.init(x: self.frame.size.width - 20, y: 2))
        
self.horizontalLayer.path = horizontalPath.cgPath
        
self.horizontalLayer.strokeColor = UIColor.white.cgColor
        
self.horizontalLayer.lineWidth = 4
        
self.horizontalLayer.cornerRadius = 2
        
self.horizontalLayer.masksToBounds = true
        
self.layer.addSublayer(self.horizontalLayer)

圆的代码基本上是相似的。

回到动画上来

点击的时候会显示周围两个圆圈显示然后消失

self.outsideAnimation = CAKeyframeAnimation.init(keyPath: "opacity")
        
self.outsideAnimation.values = [0,0.5,0.0]
        
self.outsideAnimation.keyTimes = [0,0.4,1]
        
self.outsideAnimation.autoreverses = false
        
self.outsideAnimation.duration = 0.5
        
self.outsideAnimation.timingFunctions = [CAMediaTimingFunction.init(name: kCAMediaTimingFunctionEaseOut),CAMediaTimingFunction.init(name: kCAMediaTimingFunctionDefault)]
        
self.outsideAnimation.fillMode = kCAFillModeForwards
        
self.outsideAnimation.calculationMode = kCAAnimationLinear
        
self.outsideAnimation.isRemovedOnCompletion = false

然后就执行了直线的旋转动画,如果同时添加直线的动画可能会同时在执行了,那怎么区分了,可以设置CAAnimationDelegate但是:

func animationDidStop(_ anim: CAAnimation, finished flag: Bool) {
}

anim是一个深拷贝,所以没法用self.animation == anim去区分,我这里是直接kvc区分的(大家有其他方法请告诉我):

//设置垂直动画的代理为本身以及Value
self.verticalAnimation.delegate = self
self.verticalAnimation.setValue("verticalExpandAnimation", forKey: "identifier")
self.verticalShrinkAnimation.delegate = self
self.verticalShrinkAnimation.setValue("verticalShrinkAnimation", forKey: "identifier")
//外圈圆动画
self.outsideAnimation.delegate = self
self.outsideAnimation.setValue("outsideCircleExpandAnimation", forKey: "identifier")
//内圈圆
self.insideCircleAnimation.delegate = self
self.insideCircleAnimation.setValue("insideCircleExpandAnimation", forKey: "identifier")
self.insideCircleShrinkAnimation.delegate = self
self.insideCircleShrinkAnimation.setValue("InsideshrinkAnimation", forKey: "identifier")

然后代理里面就可以直接区别开了:

switch anim.value(forKey: "identifier") as! String{
  
}

直线的动画是旋转,使用BasicAnimation就行:

//垂直线动画
self.verticalAnimation = CABasicAnimation.init(keyPath: "transform.rotation.z")
        
self.verticalAnimation.fromValue = angle(value: 0)
        
self.verticalAnimation.toValue = angle(value: 450.0)
        
self.verticalAnimation.repeatCount = 0.0
        
self.verticalAnimation.autoreverses = false
        
self.verticalAnimation.duration = 1
        
self.verticalAnimation.fillMode = kCAFillModeForwards
        
self.verticalAnimation.isRemovedOnCompletion = false
//水平线动画
self.horizontalAnimation = CABasicAnimation.init(keyPath: "transform.rotation.z")
        
self.horizontalAnimation.fromValue = angle(value: 0)
        
self.horizontalAnimation.toValue = angle(value: 360.0)
        
self.horizontalAnimation.repeatCount = 0.0
        
self.horizontalAnimation.autoreverses = false
        
self.horizontalAnimation.duration = 0.75
        
self.horizontalAnimation.fillMode = kCAFillModeForwards
        
self.horizontalAnimation.isRemovedOnCompletion = false

直线动画完成之后,圆就会扩大到整个屏幕,这个改变只需要对layer.path做动画就行:

self.insideCircleAnimation = CAKeyframeAnimation.init(keyPath: "path")
self.insideCircleAnimation.values = [
  UIBezierPath.init(arcCenter: CGPoint.init(x: self.insideCircleLayer.frame.size.width / 2, y:self.insideCircleLayer.frame.size.width / 2), radius: self.insideCircleLayer.frame.size.width / 2, startAngle: 0, endAngle: CGFloat(M_PI * 2), clockwise: true).cgPath,
  UIBezierPath.init(arcCenter: CGPoint.init(x: self.insideCircleLayer.frame.size.width / 2, y:self.insideCircleLayer.frame.size.width / 2), radius: self.insideCircleLayer.frame.size.width / 2 - 3, startAngle: 0, endAngle: CGFloat(M_PI * 2), clockwise: true).cgPath,
  UIBezierPath.init(arcCenter: CGPoint.init(x: self.insideCircleLayer.frame.size.width / 2, y:self.insideCircleLayer.frame.size.width / 2), radius: 4500, startAngle: 0, endAngle: CGFloat(M_PI * 2), clockwise: true).cgPath]

self.insideCircleAnimation.keyTimes = [0,0.4,1]
        
self.insideCircleAnimation.autoreverses = false
        
self.insideCircleAnimation.duration = 1
        
self.insideCircleAnimation.timingFunctions = [CAMediaTimingFunction.init(name: kCAMediaTimingFunctionEaseOut),CAMediaTimingFunction.init(name: kCAMediaTimingFunctionDefault)]
        
self.insideCircleAnimation.fillMode = kCAFillModeForwards
        
self.insideCircleAnimation.calculationMode = kCAAnimationLinear
        
self.insideCircleAnimation.isRemovedOnCompletion = false

最后的path我将半径设置为了4500,这个值其实只需要设置覆盖到全屏幕就行。

这就是前半部分的所有动画,然后就要执行后半部分的动画了。

全部展开后,看到一条线,然后类似于画布似的展开,这个其实也是path属性的动画,选项的出现和关闭按钮的绘制,可以发现,关闭按钮的左右直线也是不同步的,所以里面的线也是两个layer绘制出来的,绘制一般都是直线,所以这里我是改变了view.transform

//设置旋转
self.closeView.transform = self.closeView.transform.rotated(by: CGFloat(angle(value: 45.0)))

线条的消失和绘制是对strokeEnd做动画,因为要有时间差,所以动画分为两个,duration不一样。

self.closeLeftLoadAnimation=CABasicAnimation.init(keyPath:"strokeEnd")

self.closeLeftLoadAnimation.fromValue=0.0

self.closeLeftLoadAnimation.toValue=1.0

self.closeLeftLoadAnimation.duration=0.5

self.closeLeftLoadAnimation.isRemovedOnCompletion=false

self.closeLeftLoadAnimation.autoreverses=false

self.closeLeftLoadAnimation.fillMode=kCAFillModeForwards

//RightLayerAnimation

self.closeRightLoadAnimation = CABasicAnimation.init(keyPath: "strokeEnd")

self.closeRightLoadAnimation.fromValue = 0.0
        
self.closeRightLoadAnimation.toValue = 1.0
        
self.closeRightLoadAnimation.duration = 0.75
        
self.closeRightLoadAnimation.isRemovedOnCompletion = false
        
self.closeRightLoadAnimation.autoreverses = false
        
self.closeRightLoadAnimation.fillMode = kCAFillModeForwards

对于strokeEndstrokeStart属性我是这样理解的。

它们的范围都是[0,1]
`strokeEnd`代表当前描绘终点的位置比上终点位置的百分比,所以值从[0,1]就是绘制出来的过程。
`strokeStart`代表当前描绘开始点位置比上终点位置的百分比,所以值从[0,1]就是消失的过程。

选中之后消失的动画就全部这些动画的反向了,动画的顺序问题我还是使用kvc去区分的,整个控件就完成啦。

使用
let boxView = MXCheckBoxView.init
(items :["some items",""],parentView: self.view)

self.boxView.delegate = self

self.boxView.show()

实现这个代理方法之后,会在点击之后调用代理(关闭不会回调)

protocolMXCheckBoxViewDelegate{

func checkBox(checkBoxView :MXCheckBoxView,didSelect row :Int)

}
完整代码

地址:GitHub

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

推荐阅读更多精彩内容

  • 在iOS中随处都可以看到绚丽的动画效果,实现这些动画的过程并不复杂,今天将带大家一窥ios动画全貌。在这里你可以看...
    每天刷两次牙阅读 8,429评论 6 30
  • 转载:http://www.jianshu.com/p/32fcadd12108 每个UIView有一个伙伴称为l...
    F麦子阅读 6,125评论 0 13
  • 在iOS中随处都可以看到绚丽的动画效果,实现这些动画的过程并不复杂,今天将带大家一窥iOS动画全貌。在这里你可以看...
    F麦子阅读 5,075评论 5 13
  • 每个UIView有一个伙伴称为layer,一个CALayer。UIView实际上并没有把自己画到屏幕上;它绘制本身...
    shenzhenboy阅读 3,077评论 0 17
  • 处理负面情绪的最优方法 工作后,常常是好多同事在一个办公室工作,常常碰到同事抱怨...
    舒特菲拉阅读 690评论 1 8