Swift 自定义过场动画

简介

本人最近一直在学习Swift编程,然后遇到自定义过场动画的问题,再次做一个小小的总结,也希望能给同时也在学习Swift的朋友一些帮助。如果想了解OC的自定义过场动画,推荐阅读 iOS自定义转场动画/UIPresentationController,讲解的非常细致,而且有demo代码。

Demo 下载
如果不想下载请看文章最后的全部代码

第一部分

首先我们需要创建一个CustomPresentationManager类

成员变量

    //用于指定presentationController的frame
    var presentedFrame = CGRect.zero
    
    //用于标记presentation视图是否已经显示
    var isPresented = false

协议

CustomPresentationManager类需要继承两个协议分别是:UIViewControllerTransitioningDelegate
UIViewControllerAnimatedTransitioning

在UIViewControllerTransitionDelegate中需要实现三个方法,分别是
1.presentationController()

func presentationController(forPresented presented: UIViewController, presenting: UIViewController?, source: UIViewController) -> UIPresentationController?{
        //创建presentationController
        let dialog = DialogPresentationController.init(presentedViewController: presented, presenting: presenting)
        
        //设置Controller中成员变量的值
        dialog.presentedFrame = presentedFrame
        
        //返回指定的presentationController
        return dialog
    }

这个函数主要用于指定PresentationController
2.animationController(forPresented presented: UIViewController, presenting: UIViewController, source: UIViewController) -> UIViewControllerAnimatedTransitioning?

func animationController(forPresented presented: UIViewController, presenting: UIViewController, source: UIViewController) -> UIViewControllerAnimatedTransitioning?{
        isPresented = false
        NSLog("Show")
        return self
    }

这个函数会在presented视图出现的时候被调用, 在此函数中可以进行一些需要在presented视图出现之后进行的操作。 需要注意的是在这个函数中需要把isPresented置为false,原因稍后说明。

  1. animationController(forDismissed dismissed: UIViewController) -> UIViewControllerAnimatedTransitioning?
func animationController(forDismissed dismissed: UIViewController) -> UIViewControllerAnimatedTransitioning?{
        isPresented = true
        NSLog("Disappear")
        return self
    }

这个函数的作用正好和上一个相反,在presentation视图消失的时候会被调用,并在此函数中把isPresented置为false。


UIViewControllerAnimatedTransitioning协议主要用于指定transitioning的时间和动画的细节,其中需要实现两个方法。

  1. transitionDuration(using transitionContext: UIViewControllerContextTransitioning?) -> TimeInterval
func transitionDuration(using transitionContext: UIViewControllerContextTransitioning?) -> TimeInterval{
        return 0.8
    }

这个函数并没有什么好说的,需要返回transitioning 的时间。

  1. animateTransition(using transitionContext: UIViewControllerContextTransitioning)
func animateTransition(using transitionContext: UIViewControllerContextTransitioning){
        if isPresented{
            disappear(transitionContext: transitionContext)
        }else{
            showUp(transitionContext: transitionContext)
        }
    }

这个函数需要注意的地方是,在transition开始和结束的时候,该函数都会被调用,所以这也是上文提到的isPresented变量需要被用到的地方。需要根据isPresented来判断Presentation视图是否展示,从而指定不同的transition方式。值得一提的是transitionContext这个变量,transition所需要的东西全部都包含在这个变量里面。showUp和disappear函数指定了具体的transition方式,具体细节请参考上文中的Demo代码。

第二部分

这个部分主要介绍自定义的UIPresentationController里面的一些细节和注意点。
这部分的代码比较少而且很简单,就全部粘出来吧。

import UIKit

class DialogPresentationController: UIPresentationController {
    var presentedFrame = CGRect.zero
    var isLayout = false
    override func containerViewDidLayoutSubviews(){
        if !isLayout {
            presentedView?.frame = presentedFrame
            
            containerView?.insertSubview(coverbutton, at: 0)
            
            coverbutton.addTarget(self, action: #selector(closeDialog), for: .touchUpInside)
            isLayout = true
        }
    }
    
    func closeDialog(){
        presentedViewController.dismiss(animated: true, completion: nil)
    }
    private lazy var coverbutton: UIButton = {
        let btn = UIButton.init(frame: UIScreen.main.bounds)
        btn.backgroundColor = UIColor.lightGray
        btn.alpha = 0.8
        return btn
    }()
}

最需要注意的是containerViewDidLayoutSubviews方法在presented视图展示和消失的时候都会被调用。如果不用isLayout判断是否layout过就会重复layout,导致一些奇奇怪怪的错误。

全部代码:

CustomTransitioningManager

//
//  CustomTransition.swift
//  AnimationTest
//
//  Created by Albert-z on 2017-05-14.
//  Copyright © 2017 Albert-z. All rights reserved.
//

import UIKit

class CustomTransitioningManager: NSObject, UIViewControllerTransitioningDelegate,UIViewControllerAnimatedTransitioning {
    
    //用于指定presentationController的frame
    var presentedFrame = CGRect.zero
    
    //用于标记presentation视图是否已经显示
    var isPresented = false
    
    //MARK:UIViewControllerAnimatedTransitioning
    func transitionDuration(using transitionContext: UIViewControllerContextTransitioning?) -> TimeInterval{
        return 0.8
    }
    
    
    func animateTransition(using transitionContext: UIViewControllerContextTransitioning){
        if isPresented{
            disappear(transitionContext: transitionContext)
        }else{
            showUp(transitionContext: transitionContext)
        }
    }
    
    
    //MARK:UIViewControllerTransitioningDelegate
    
    func presentationController(forPresented presented: UIViewController, presenting: UIViewController?, source: UIViewController) -> UIPresentationController?{
        //创建presentationController
        let dialog = DialogPresentationController.init(presentedViewController: presented, presenting: presenting)
        //设置Controller中成员变量的值
        dialog.presentedFrame = presentedFrame
        
        //返回指定的presentationController
        return dialog
    }

    func animationController(forPresented presented: UIViewController, presenting: UIViewController, source: UIViewController) -> UIViewControllerAnimatedTransitioning?{
        isPresented = false
        NSLog("Show")
        return self
    }
    
    func animationController(forDismissed dismissed: UIViewController) -> UIViewControllerAnimatedTransitioning?{
        isPresented = true
        NSLog("Disappear")
        return self
    }
    
    //MARK:Custom func
    func showUp(transitionContext: UIViewControllerContextTransitioning){
        let view = transitionContext.view(forKey: .to)
        
        transitionContext.containerView.addSubview(view!)
        
        view?.transform = .init(scaleX: 0.0, y: 0.0)
        
        view?.layer.anchorPoint = .init(x: 0.5, y: 0.5)
        
        UIView.animate(withDuration: 0.5, animations: { 
            view?.transform = .identity
        }) { (_) in
            transitionContext.completeTransition(true)
        }
        
    }
    
    func disappear(transitionContext: UIViewControllerContextTransitioning){
        
        let view = transitionContext.view(forKey: .from)
        
        view?.transform = .init(scaleX: 1.0, y: 1.0)
    
        UIView.animate(withDuration: 0.5, animations: { 
            view?.transform = .init(scaleX: 1.0, y: 0.001)
        }) { (_) in
            transitionContext.completeTransition(true)
        }
        
    }
}

DialogPresentationController

//
//  DialogPresentationController.swift
//  AnimationTest
//
//  Created by Albert-z on 2017-05-14.
//  Copyright © 2017 Albert-z. All rights reserved.
//

import UIKit

class DialogPresentationController: UIPresentationController {
    var presentedFrame = CGRect.zero
    var isLayout = false
    override func containerViewDidLayoutSubviews(){
        if !isLayout {
            presentedView?.frame = presentedFrame
            
            containerView?.insertSubview(coverbutton, at: 0)
            
            coverbutton.addTarget(self, action: #selector(closeDialog), for: .touchUpInside)
            isLayout = true
        }
    }
    
    func closeDialog(){
        presentedViewController.dismiss(animated: true, completion: nil)
    }
    private lazy var coverbutton: UIButton = {
        let btn = UIButton.init(frame: UIScreen.main.bounds)
        btn.backgroundColor = UIColor.lightGray
        btn.alpha = 0.8
        return btn
    }()
}

ViewController

//
//  ViewController.swift
//  AnimationTest
//
//  Created by Albert-z on 2017-05-13.
//  Copyright © 2017 Albert-z. All rights reserved.
//

import UIKit

class ViewController: UIViewController {

    var window: UIWindow?
    var manager:CustomTransitioningManager?
    override func viewDidLoad() {
        super.viewDidLoad()
        
    }

    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }

    @IBAction func btnClick(_ sender: Any) {
        manager = CustomTransitioningManager.init()
        manager?.presentedFrame = CGRect.init(x: UIScreen.main.bounds.width/2 - 150.0, y: UIScreen.main.bounds.height/2 - 100.0, width: 300, height: 150)
        let sb = UIStoryboard.init(name: "DialogVC", bundle: nil)
        let dialogVC = sb.instantiateInitialViewController()
        dialogVC?.modalPresentationStyle = .custom
        dialogVC?.transitioningDelegate = manager
        present(dialogVC!, animated: true, completion: nil)
    }

}


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

推荐阅读更多精彩内容