1: 導航欄隱藏之間跳轉 & 導航欄隱藏后顯示自定義導航欄
(由于系統導航欄透明造成的動畫問題 暫時還未找到好的辦法,目前都是采用隱藏 添加自定義導航欄。
之前對于導航欄隱藏和透明的VC ,采用了一個封裝的導航欄VIew 但是需要單獨初始化,顯的麻煩,
因為隱藏的導航欄,大部分都還是需要一個導航條的,只是樣式跟整體風格不同,因此也將這個自定的導航條 改為 runtime 方式,
iOS導航欄對于隱藏造成的動畫 可以通過在viewWillAppear和viewWillDisappear 來設置完成
- (void)viewWillAppear:(BOOL)animated {
[super viewWillAppear:animated];
[self.navigationController setNavigationBarHidden:YES animated:animated];
}
- (void)viewWillDisappear:(BOOL)animated {
[super viewWillDisappear:animated];
[self.navigationController setNavigationBarHidden:NO animated:animated];
}
但是對于多級跳轉的則不行,之前一開始 自己采用的是繼承,但是繼承不是對于整個項目所適用的,而實際中繼承的獨特,也只是為了判斷當前控制器 是否需要隱藏導航欄,之前一開始是通過類型的方式判斷的,
+ (BOOL)naviIsHiddenWithVC: (UIViewController *)vc {
if ([vc isKindOfClass:[NaviHiddenController class]]) {
return YES;
}
return NO;
}
這樣的判斷 實際只需要通過caretory添加一個屬性或者方法即可達到同樣效果 判斷出是否需要隱藏。
具體代碼:
1、我采用的是用實例方法 來標識 是否需要隱藏(也可以改為實例對象,調用的方式)
// 用于判斷當前控制器是否需要隱藏導航欄 默認不需要隱藏 需要隱藏的控制器 只需要重寫即可
- (BOOL)wyj_naviBarIsHidden {
return NO;
}
2、交換方法
#pragma mark - 交換相關方法 ,也是為了在
+ (void)load {
method_exchangeImplementations(class_getInstanceMethod(self, @selector(viewWillAppear:)),
class_getInstanceMethod(self, @selector(wyjBarHidden_viewWillAppear:)));
method_exchangeImplementations(class_getInstanceMethod(self, @selector(viewWillDisappear:)),
class_getInstanceMethod(self, @selector(wyjBarHidden_viewWillDisappear:)));
// 用于設置自定義導航欄View的
method_exchangeImplementations(class_getInstanceMethod(self, @selector(viewDidLoad)),
class_getInstanceMethod(self, @selector(wyjBarHidden_viewDidLoad)));
method_exchangeImplementations(class_getInstanceMethod(self, @selector(setTitle:)),
class_getInstanceMethod(self, @selector(wyjBarHidden_setTitle:)));
}
3、主要是push或者pop前后 的VC是否隱藏,判斷對才能確保相互跳轉之間的動畫不會出問題
#pragma mark - super
- (void)wyjBarHidden_viewWillAppear:(BOOL)animated {
[self wyjBarHidden_viewWillAppear:animated];
if ([self wyj_naviBarIsHidden]) {
[self.navigationController setNavigationBarHidden:YES animated:animated];
}
// 用于自定義導航欄的
[self.view bringSubviewToFront:self.wyj_naviView];
}
- (void)wyjBarHidden_viewWillDisappear:(BOOL)animated {
[self wyjBarHidden_viewWillDisappear:animated];
if ([self wyj_naviBarIsHidden] == NO) {
return;
}
int pushNext = [self pushNextOrPop];
// push下一個
if (pushNext == 1) {
if (NO == [self viewControllersPushIsHidden]) {
[self.navigationController setNavigationBarHidden:NO animated:animated];
}
return;
}
// not Push
if (NO == [self viewControllersPopIsHidden]) {
[self.navigationController setNavigationBarHidden:NO animated:animated];
}
}
// push : 1
// pop : 0
- (int)pushNextOrPop {
NSArray *viewControllers = self.navigationController.viewControllers;
if (viewControllers.count > 1 && [viewControllers objectAtIndex:viewControllers.count-2] == self) {
return 1; // push
}
return 0;
}
// 此處判斷是否時隱藏的控制器 需要判斷涉及的相關類
// push下一個 是不是隱藏
- (BOOL)viewControllersPushIsHidden {
NSArray *viewControllers = self.navigationController.viewControllers;
UIViewController *vc = [viewControllers objectAtIndex:viewControllers.count-1];
return [vc wyj_naviBarIsHidden];
}
// pop last上一個
- (BOOL)viewControllersPopIsHidden {
NSArray *viewControllers = self.navigationController.viewControllers;
UIViewController *vc = [viewControllers objectAtIndex:viewControllers.count-1];
return [vc wyj_naviBarIsHidden];
}
下面是添加的自定義導航欄的
1、有一個保存導航View的對象,懶加載
2、在viewDidLoad中 添加,所以需要runtime交換方法,
3、在viewWIllAppear中 將自定義導航顯示在最前面
4、由于title的時機不確定,所以在重寫title,在使用的可以對title顯示,這樣對于一些需要導航欄透明的 只需要實現隱藏導航欄方法即可,不需要在寫任何代碼
// ================== 添加默認需要的自定義導航欄 ========================
#pragma mark - 添加默認需要的自定義導航欄
- (void)wyjBarHidden_viewDidLoad {
[self wyjBarHidden_viewDidLoad];
[self.view addSubview:self.wyj_naviView];
}
- (void)wyjBarHidden_setTitle:(NSString *)title {
[self wyjBarHidden_setTitle:title];
self.wyj_naviView.titleLabel.text = title;
}
#pragma mark get set wyj_naviView
- (NavigationImitateView *)wyj_naviView {
if (objc_getAssociatedObject(self, _cmd) == nil) {
if ([self wyj_naviBarIsHidden]) {
[self wyj_NaviViewInit];
}
}
return objc_getAssociatedObject(self, _cmd);
}
- (void)setWyj_naviView:(NavigationImitateView *)wyj_naviView {
objc_setAssociatedObject(self, @selector(wyj_naviView), wyj_naviView, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- (void)wyj_naviBackAction {
[self.navigationController popViewControllerAnimated:YES];
}
- (void)wyj_NaviViewInit {
NavigationImitateView *view = [NavigationImitateView initDefaule];
view.titleLabel.text = self.title;
[view.leftButton addTarget:self action:@selector(wyj_naviBackAction) forControlEvents:(UIControlEventTouchUpInside)];
self.wyj_naviView = view;
}
側滑返回手勢
在上面隱藏導航欄后 ,系統的側滑返回手勢就失效了,因此我們添加一個側滑返回手勢
因為我的導航欄是統一繼承的,所以我是在導航欄的基類里面寫的,畢竟項目中 導航就tabbar那么幾個
添加自定義導航欄 ,并禁止系統的
- (void)setCustomGestureRecognizer {
// 獲取系統自帶滑動手勢的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;
}
// 判斷當前是否禁止側滑返回,
UIViewController *topViewController = self.childViewControllers.lastObject;
if ([topViewController wyj_naviPopGRDisable]) {
return NO;
}
// ---------------------- return YES------------------------------
//如果在此處 return YES ,則是全屏側滑返回
# 需要注意的是:
# 全屏返回手勢,會和 系統tabbarCell 左滑刪除的時候 手勢沖突,導致左滑刪除不出來,
# 簡單點我是將當前手勢改為 左滑邊緣處 才能夠觸發,基本和系統的一樣
CGPoint location = [gestureRecognizer locationInView:self.view];
CGPoint offSet = [gestureRecognizer locationInView:gestureRecognizer.view];
BOOL result = (0 < offSet.x && location.x <= 40);
return result;
// return YES;
}
如果想要全屏側滑返回手勢 和 tableViewCell 左滑手勢共存,則需要提供額外接口,因為需要將兩個的代理結合,判斷需要如何處理哪個優先級,此處用上述交簡單的, 也算是模仿系統的方式處理
具體代碼我已經抽出來
//使用的時候 只需要引入 #import "WYJNavigation.h"
//---------需要隱藏導航欄的文件 -----------
// 重寫
- (BOOL)wyj_naviBarIsHidden {
return YES;
}
//---------需要添加左側滑動返回的 -----------
讓根導航繼承 WYJNavigationController
文件下載:
鏈接: https://pan.baidu.com/s/1wdUN5Wa7jvgDcfiH0GTMgA 密碼: 73un