8行代碼教你搞定導航控制器全屏滑動返回效果

原文地址:啊崢的簡書

前言

此次文章,講述的是導航控制器全屏滑動返回效果,而且代碼量非常少,10行內搞定。
效果如圖:


效果.gif

如果喜歡我的文章,可以關注我,也可以來小碼哥,瞭解下我們的iOS培訓課程。陸續還會有更新ing....

一、自定義導航控制器

目的:以後需要使用全屏滑動返回功能,就使用自己定義的導航控制器。

二、分析導航控制器側滑功能

效果:導航控制器默認自帶了側滑功能,當用戶在界面的左邊滑動的時候,就會有側滑功能。

系統自帶的側滑效果:


側滑.gif

分析:
1.導航控制器的view自帶了滑動手勢,只不過手勢的觸發範圍只能在左邊。

2.當用戶在界面左邊拖動,就會觸發滑動手勢方法,並且有滑動返回功能,說明系統手勢觸發的方法已經實現了滑動返回功能。

3.爲什麼說系統手勢觸發的方法已經實現了滑動返回功能?

原因:

  • 創建滑動手勢對象的時候,需要綁定監聽者,當觸發手勢的時候會調用target的action。
UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:target action:action];
  • 當用戶在界面左邊滑動,有滑動返回功能,這是因爲觸發手勢了,調用target的action方法,說明action方法內部實現滑動返回功能,否則就不會有滑動返回效果。

三、實現全屏滑動功能分析

  • 打印導航控制器自帶的滑動手勢,看下它的真實面目。
  • 系統自帶的滑動手勢interactivePopGestureRecognizer
//  self指向的導航控制器,在導航控制器的viewDidLoad方法打印
- (void)viewDidLoad {
    [super viewDidLoad];
    NSLog(@"%@",self.interactivePopGestureRecognizer);
}

打印結果圖片:


Snip20150801_1.png
  • 由圖中可知:
  • 1.系統自帶的手勢是UIScreenEdgePanGestureRecognizer類型對象,屏幕邊緣滑動手勢
  • 3.系統自帶手勢target是_UINavigationInteractiveTransition類型的對象
  • 4.target調用的action方法名叫handleNavigationTransition:

分析:

  • UIScreenEdgePanGestureRecognizer,看名稱就知道,這個手勢的範圍只能在屏幕的周邊,就是因爲這個手勢,系統自帶的滑動效果,只能實現側邊滑動。

四、如何實現全屏滑動功能

  • 給自己的導航控制器,添加一個全屏的滑動手勢,調用系統自帶滑動手勢的targetaction方法,利用系統實現的滑動返回功能,加上自己全屏滑動手勢,就有全屏滑動功能了。

  • 問題:如何拿到系統自帶的target對象?,action方法名已經知道,而且系統肯定在target對象實現了,只要拿到target對象,調用這個方法就行。

  • 通過打印系統自帶的滑動手勢的代理,發現正好是_UINavigationInteractiveTransition對象,因此我猜測這個代理對象就是target對象,只要拿到它,就拿到系統自帶滑動手勢的target對象。
    // 打印系統自帶滑動手勢的代理對象
    NSLog(@"%@",self.interactivePopGestureRecognizer.delegate);
  • 打印圖片:

Snip20150801_2.png
  • 導航控制器全屏滑動注意點:
  • 1.禁止系統自帶滑動手勢使用
  • 2.只有導航控制器的非根控制器才需要觸發手勢,使用手勢代理,控制手勢觸發。

  • 全屏滑動代碼實現

- (void)viewDidLoad {
    [super viewDidLoad];

    // 獲取系統自帶滑動手勢的target對象
    id target = self.interactivePopGestureRecognizer.delegate;

    // 創建全屏滑動手勢,調用系統自帶滑動手勢的target的action方法
    UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:target action:@selector(handleNavigationTransition:)];

    // 設置手勢代理,攔截手勢觸發
    pan.delegate = self;

    // 給導航控制器的view添加全屏滑動手勢
    [self.view addGestureRecognizer:pan];

    // 禁止使用系統自帶的滑動手勢
    self.interactivePopGestureRecognizer.enabled = NO;

}

// 什麼時候調用:每次觸發手勢之前都會詢問下代理,是否觸發。
// 作用:攔截手勢觸發
- (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer
{
    // 注意:只有非根控制器纔有滑動返回功能,根控制器沒有。
    // 判斷導航控制器是否只有一個子控制器,如果只有一個子控制器,肯定是根控制器
    if (self.childViewControllers.count == 1) {
        // 表示用戶在根控制器界面,就不需要觸發滑動手勢,
        return NO;
    }
    return YES;
}


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章