輕鬆學習之二——iOS利用Runtime自定義控制器POP手勢動畫

前言

蘋果在IOS7之後給導航控制器增長了一個Pop的手勢,只要手指在屏幕邊緣滑動,當前的控制器的視圖就會跟隨你的手指移動,當用戶鬆手後,系統會判斷手指拖動出來的大小來決定是否要執行控制器的Pop操做。git

nav_pop_origin.gifgithub

這個操做的想法很是好,可是系統給咱們規定的範圍必須是屏幕左側邊緣才能夠觸發,這樣實際使用過程當中對於有些產品會產生不便,因而有些app就採起整個屏幕都響應這個手勢而且pop動畫仍是用系統原生的,這樣操做起來確實方便好多。數組

nav_pop_custom.gifapp

開始你們必定會有疑問,給控制器的View加個手勢而後拖動控制器的View時改變它的frame不就能夠了嗎?沒錯,加手勢這個想法是正確的。可是,由咱們本身來改變控制器視圖的位置是比較麻煩的,細心的朋友必定發現了,咱們自定義pop手勢上面的導航欄也是在隨着你的手勢拖拽而變更的,因此這樣作還須要負責導航欄的動畫,並且有一個重點問題,若是單獨拖動view,這個view下面會是黑黑的一片,由於控制器的push和pop層級是由系統管理的。ide

nav_pop_failed.gif學習

因此走這條路雖然能夠,但實現起來會比較艱辛。那麼,如何實現這個效果呢?今天就給你們提供兩套實現方案。優化


[1]動畫

方案一:自定義UIViewControllerInteractiveTransitioning對象,實現導航控制器代理方法。

這個是蘋果官方推薦的作法,在WWDC 2013 218 - Custom Transitions Using View Controllers中有說明。spa

這套方案雖然實現比較麻煩,可是動畫相對靈活,你能夠實現這樣的效果,設計

nav_pop_cube.gif

也能夠有這種效果。

nav_pop_flip.gif

其實這個拖動過程屬於導航控制器的動畫,因此咱們須要重寫UINavigationController的兩個代理方法,navigationController:animationControllerForOperation:fromViewController:toViewController:(名字很長下面就稱爲方法1)和
navigationController:interactionControllerForAnimationController:(方法2)。
解釋一下他們的做用,方法1是蘋果提供給咱們用來重寫控制器之間轉場動畫的(pop或者push)。方法2你能夠這樣理解,蘋果讓咱們返回一個交互的對象,用來實時管理控制器之間轉場動畫的完成度,經過它咱們可讓控制器的轉場動畫與用戶交互(注意一點,若是方法1返回是nil,方法2是不會調用的,也就是說,只有咱們自定義的動畫才能夠與控制器交互)。

下面咱們來看一下實現過程。爲了便於你們理解,我會盡可能在Demo中的註釋寫的最清晰明瞭。
同時,咱們先用最簡單的代碼實現,在這篇文章的最後我會對本例中的Demo提供一個相對合理的寫法。

首先在方法1中,咱們返回一個遵照了UIViewControllerAnimatedTransitioning協議的對象,它就是自定義的動畫對象,咱們給它起名PopAnimation,在這個類中實現兩個方法來自定義轉場動畫。

屏幕快照 2015-03-28 下午6.49.05.png

再來看方法2,咱們須要返回一個遵照了UIViewControllerInteractiveTransitioning協議的對象(提示一下,這兩個協議容易混淆,要注意區分,一個是負責動畫,一個是負責交互過程),蘋果已經有一個類專門處理這個功能,它叫UIPercentDrivenInteractiveTransition,固然你也能夠自定義一個這樣的類。咱們能夠這樣理解它的做用:前面在方法1中返回的動畫,會在執行的過程當中被系統分解以用於用戶交互,這個交互過程的動畫完成度就由它來調控。下面咱們來看一下如何使用它。(爲了讓控制器視圖拖動,咱們給控制器的視圖加了一個拖動手勢,在拖動方法裏咱們對這個對象進行操做)

屏幕快照 2015-03-29 下午12.33.59.png

最後在視圖控制器裏重寫導航欄的兩個方法。

屏幕快照 2015-03-29 下午12.37.51.png

有兩點不要忘記:

  1. 設置導航控制器的代理爲當前控制器。
  2. 給控制器加手勢。

OK,這樣咱們就完成了這個過程。

nav_pop_own.gif


[2]

方案二:Runtime+KVC

要了解這樣的作法,須要有Runtime的一些知識,會涉及到私有變量、私有方法的獲取,可是這樣作比較簡單也比較有趣,若是你感興趣就繼續看下去吧。關於Runtime的知識,從此我會分享到博客裏,朋友們敬請期待。

爲了方便你們閱讀下面的代碼,咱們須要先了解系統的這個手勢。

前面咱們瞭解到,這個手勢屬於UINavigationController,咱們就跳到它的頭文件裏看看能不能找到線索。這個思路是正確的,確實有一個手勢叫作interactivePopGestureRecognizer。屬性爲readonly,就是說咱們不能給他換成自定義的手勢,可是能夠設置enable=NO。ok,既然找到了它,就打印一下看看它究竟是一個什麼手勢。

屏幕快照 2015-03-26 下午5.17.35.png

經過log,咱們看到他屬於UIScreenEdgePanGestureRecognizer這個類(以前我是沒有用到過),它繼承自UIPanGestureRecognizer,出如今IOS7之後,是專門處理在屏幕邊緣觸發的手勢類型,而且只有一個屬性叫edges,用來設置它的觸發邊緣(上、下、左、右、所有)。看到這裏一些朋友會想,直接改它的edges爲所有可不能夠?通過試驗瞭解到,改這個屬性是沒用的,它只能用來觸發邊緣,設爲所有的意思是四個方向的邊緣會觸發,並且用來作控制器POP手勢的只有左邊緣。

咱們繼續看它的log。控制檯除了打印了它的類,還打印了它的觸發target:_UINavigationInteractiveTransition(這是一個私有類,看來是專門用來作導航控制器交互動畫的),和action:handleNavigationTransition(這是它的一個私有方法),咱們要作的就是新建一個UIPanGestureRecognizer,讓它的觸發和系統的這個手勢相同,這就須要利用runtime獲取系統手勢的target和action。

那麼如何獲取這個target呢?一開始我用kvc想直接獲取這個手勢的target,程序崩潰了,原來它根本沒有這樣一個屬性。因此我能想到的是,先利用runtime遍歷它的全部成員變量,看看系統是怎麼存儲這個屬性的,

屏幕快照 2015-03-29 下午3.25.02.png


經過log咱們能夠看到,UIGestureRecognizer有一個叫_targets的屬性,它的類型爲NSMutableArray。

屏幕快照 2015-03-29 下午3.25.09.png


它是用數組來存儲每個target-action,因此能夠動態的增長手勢觸發對象。那麼又是什麼存儲每個target-action呢?爲了瞭解這個咱們拿到這個屬性的名字"_targets"經過kvc獲取它,接着打印出來。

屏幕快照 2015-03-29 下午3.33.54.png

屏幕快照 2015-03-29 下午3.34.01.png

能夠看到,因爲系統重寫了它的description方法,因此咱們沒辦法經過打印獲取這個對象是什麼類型。既然不能打印,那麼咱們就用斷點調試,來看它的真實類型,

屏幕快照 2015-03-29 下午3.37.32.png

咱們看到,原來每個target-action是用UIGestureRecognizerTarget這樣一個類來存儲的,它也是一個私有類。
蘋果把許多的類作私有化也是有緣由所在,其實在平時咱們拿到這個類也是沒有用的,他們的目的之一是避免對開發者公開無用的類,影響了封裝性。因此在類的設計上,仍是要向蘋果學習。

下面直接看代碼。

咱們在控制器的ViewDidLoad加上這段代碼,而且它只須要執行一次。

屏幕快照 2015-03-29 下午4.07.48.png


優化

這個demo我會提供給你們,下面簡單說下程序的優化思路。

  • 優化點一:對於方案一,其實不該該把導航控制器的代理方法以及手勢處理的方法交給視圖控制器,由於這段代碼不是屬於某一個視圖控制器,而是全局的導航控制器,因此咱們應該參考蘋果的設計思想:新建一個專門管理交互過程的對象,這個類咱們叫作NavigationInteractiveTransition。

  • 優化點二:再來看以前的ViewDidLoad中只執行一次的代碼,其實寫在這裏也不夠穩當,一樣的,這段代碼也不屬於某一個Controller,優化方案是新建一個導航控制器,在這個導航控制器的viewDidLoad中寫上這些代碼,這樣也並不須要dispatch once。

  • 優化點三:因爲咱們自定義的手勢是加在一個私有view上,這個view是一個全局的,因此當這個控制器爲根控制器時,咱們的手勢仍是在起做用,這就至關於對根控制器作了pop操做,這會出現一個錯誤nested pop animation can result in corrupted navigation bar。致使這個錯誤的緣由還有一個,若是咱們pop的動畫正在執行,再去觸發一次手勢,會致使導航控制器和導航條的動畫混亂。爲了不問題出現咱們須要成爲手勢的代理,判斷當前控制器是否爲根控制器而且pop或者push動畫是否在執行(這個變量是私有的,須要用kvc來獲取)。

    屏幕快照 2015-03-30 下午5.06.24.png

通過最後的優化,視圖控制器能夠什麼都不寫,想使用這個效果,只要使用咱們自定義的導航控制器就能夠了,這樣的好處是手勢動畫與控制器徹底解耦,而且不用給每個控制器都addGesture。


給你們推薦一個倉庫https://github.com/nst/iOS-Runtime-Headers,這個倉庫能夠調取蘋果的全部私有方法頭文件,至關強大。

最後放上這個demo的地址:https://github.com/zys456465111/CustomPopAnimation(使用時,切換工程的scheme就能切換不一樣方案。對於方案二,只須要導航控制器的類就能夠了。)

感謝你們,輕鬆學習系列還會繼續下去,我會盡可能寫出更多通俗易懂的文章,讓開發變得輕鬆起來,個人微博:http://weibo.com/JazysYu


 

文/J_雨(簡書做者) 原文連接:http://www.jianshu.com/p/d39f7d22db6c 著做權歸做者全部,轉載請聯繫做者得到受權,並標註「簡書做者」。

相關文章
相關標籤/搜索