ios框架iOS 开发随笔我的ios进阶

基于自定义url实现万能导航的探索(一)

2016-03-26  本文已影响656人  __Null

NSURL是一个较为常用的类,通常用一个已知的字符串来初始化,例如用@"https://www.baidu.com" 来初始化:

NSURL *url = [NSURL URLWithString:@"https://www.baidu.com"];

通常这个字符串对象会包含url访问的资源路径以及相关参数。那么自定义url也会遵守相应的规则,比如自定义url,字符串为

@"navigateApp://action?navigationKey=personalcenter"。

今天要讲的显然不是如何初始化NSURL对象和自定义NSURL,但是今天要讲的确实基于NSURL的两个属性值的,一个是NSURL.scheme,另一个是NSURL.query。如上字符串的scheme=@"navigateApp",query=@"navigationKey=personalcenter"。这样我们就很清楚的看到了,scheme定义了我们要处理的url的类型,query包含了参数配置,我们可以通过一定的方法把query解析成NSDictionary对象,更方便我们的使用。我们讲NSURL.query作为参数传入如下方法(类方法),即可拿到最终的NSDictionary *param = @{@"navigationKey":@"perosnalcenter"};

+ (NSMutableDictionary *)dictionaryWithQuery:(NSString *)query{
    if(query.length == 0){
        return nil;
    }
    NSArray *arrKeyValue = [query componentsSeparatedByString:@"&"];
    if(!arrKeyValue ||!arrKeyValue.count){
        return nil;
    }
    NSMutableDictionary *dicKeyValue = [[NSMutableDictionary alloc] initWithCapacity:5];
    for(NSInteger loop = 0; loop < arrKeyValue.count; loop++){
        NSString *keyValue = [arrKeyValue objectAtIndex:loop];
        if(keyValue.length == 0){
            continue;
        }
        
        NSArray *arr_KeyValue = [keyValue componentsSeparatedByString:@"="];
        if(!arr_KeyValue || arr_KeyValue.count != 2){
            continue;
        }
        
        NSString *key = [arr_KeyValue objectAtIndex:0];
        NSString *value = [arr_KeyValue objectAtIndex:1];
        
        if(key.length > 0 && value.length > 0){
            [dicKeyValue setObject:[value stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding] forKey:key];
        }
    }
    return dicKeyValue;
}

我们已经拿到url的参数,这样是不是很神奇呢?No,No,No。仅仅理解了这些,并不能发挥url的功能。要想发挥自定义url的功能我们需要运用的场景并且定义url的规则。

如下我们定义一种url的使用场景及url的定义规则:
我们通过url来做应用内导航控制器的导航:通过url的navigationKey的value确定导航到对应的UIView Controller,通过其他的参数来向该控制器的实例传值。假如我们处理的url是以@"navigateApp://action?“开头的,那么在query部分navigationKey将成为必须节点,其他传值的节点根据控制器的外部公开的属性来定义。

假如我们要实现如下url的跳转.建议写在工具类的.h中


基于自定义url实现万能导航的探索(一)

第一个,通过该url导航到个人中心.
第二个,通过该url导航到id= _ordereid的订单详情页面,原则上跳转到的控制器要公开定义orderid属性,以便外部赋值。

首先我们来看HKWHNavigator.h文件的相关方法声明。他的采用单例模式,外部久公开了两个方法,很简洁。

#import <Foundation/Foundation.h>
#import <UIKit/UIKit.h>
@interface HKWHNavigator : NSObject
+ (HKWHNavigator *)center;
- (BOOL)canNavigateInApp:(NSURL *)url;//能不能跳转
- (BOOL)nivigateInApp:(NSURL *)url;//跳转,并返回能不能跳转。
@end

对应HKWHNavigator.m文件的相关方法实现如下

#import "HKWHNavigator.h"
//引入了要跳转到的控制器的头文件。
#import "HKPersonalCenterViewController.h"
#import "HKOrderDetailViewController.h"//要定义外部赋值的属性或变量
@interface HKWHNavigator()
@property (nonatomic, retain) NSArray *arrNavigator;
- (UINavigationController *)naviController;
@end

@implementation HKWHNavigator
+ (HKWHNavigator *)center{
    static dispatch_once_t predicate;
    static HKWHNavigator *sharedInstance = nil;
    dispatch_once(&predicate, ^{
        sharedInstance = [[HKWHNavigator alloc] init];
    });
    return sharedInstance;
}

- (instancetype)init{
    if(self = [super init]){
        //表明我们只处理@“navigateApp://"开头的自定义url.
        self.arrNavigator = @[@"navigateApp"];
    }
    return self;
}

- (BOOL)canNavigateInApp:(NSURL *)url{
    if([_arrNavigator containsObject:url.scheme]){
        return YES;
    }
    return NO;
}

- (BOOL)nivigateInApp:(NSURL *)url{
    if([[HKWHNavigator center] canNavigateInApp:url]){
        NSDictionary *dicAttach = [HKWHNavigator dictionaryWithQuery:url.query];
        NSString * navigationKey = [dicAttach objectForKey:@"navigationKey"];
        
        if([navigationKey isEqualToString:@"personal"]){
            //导航到个人中心页面(ps:无需外部传参)
            HKPersonalCenterViewController *nextViewController = [[HKPersonalCenterViewController alloc] init];
            [self.naviController pushViewController:nextViewController animated:YES];
        }
        
        else if([navigationKey isEqualToString:@"orderdetail"]){
            //导航到订单详情页面(ps:需要外部传参)
            
            //(1)通过引入的控制器头文件来实例化,并导航。
            HKOrderDetailViewController *nextViewController = [[HKOrderDetailViewController alloc] init];
            nextViewController. orderid = [dicAttach objectForKey:@"orderid"];
            [self.naviController pushViewController:nextViewController animated:YES];

            //(2)也可以通过NSClassFromString方法和KVC做到传参,并导航。
            //这种方式不需要在该类中引入目标类的头文件。
            Class uiviewcontroller_class = NSClassFromString(@"HKOrderDetailViewController");
            
            id nextViewController = [[uiviewcontroller_class alloc] init];
            [nextViewController setValue:[dicAttach objectForKey:@"orderid"] forKey:@"orderid"];
            [self.naviController pushViewController:nextViewController animated:YES];
        }
        return YES;
    }
    return NO;
}

//获取当前导航控制器。
- (UINavigationController *)naviController{
    UINavigationController *sharedController = nil;
    UIWindow *window = [UIApplication sharedApplication].keyWindow;
    UIViewController *rootViewController = window.rootViewController;
    
    //找到当前的UINavigationController。
    //这会根据你App的结构不同而有所差异,这需要你分析自己应用的结构找到当前控制器。
    if([rootViewController isKindOfClass:[UITabBarController class]]){
        sharedController = (UINavigationController *)[(UITabBarController *)rootViewController selectedViewController];
    }
    else if([rootViewController isKindOfClass:[UINavigationController class]]){
        sharedController = (UINavigationController *)rootViewController;
    }
    return sharedController;
}

//解析url.query部分,返回NSDictionary格式。(按照如上的场景和规则,该返回值必然含有navigationKey节点)
+ (NSMutableDictionary *)dictionaryWithQuery:(NSString *)query{
    if(query.length == 0){
        return nil;
    }
    NSArray *arrKeyValue = [query componentsSeparatedByString:@"&"];
    if(!arrKeyValue ||!arrKeyValue.count){
        return nil;
    }
    
    NSMutableDictionary *dicKeyValue = [[NSMutableDictionary alloc] initWithCapacity:5];
    for(NSInteger loop = 0; loop < arrKeyValue.count; loop++){
        NSString *keyValue = [arrKeyValue objectAtIndex:loop];
        if(keyValue.length == 0){
            continue;
        }
        
        NSArray *arr_KeyValue = [keyValue componentsSeparatedByString:@"="];
        if(!arr_KeyValue || arr_KeyValue.count != 2){
            continue;
        }
        
        NSString *key = [arr_KeyValue objectAtIndex:0];
        NSString *value = [arr_KeyValue objectAtIndex:1];
        
        if(key.length > 0 && value.length > 0){
            [dicKeyValue setObject:[value stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding] forKey:key];
        }
    }
    return dicKeyValue;
}
@end

这样内部导航的机制基本实现了,那么外部又该怎么调用呢?下面请看

//导航到个人中心
//这种有可变参数的考验是用宏来定义,更方便。
[[HKWHNavigator center] nivigateInApp: KNavigate_PersonalCenter];

//导航到订单号为2015032688的订单性情页面
[[HKWHNavigator center] nivigateInApp: KNavigate_OrderDetail(@"2015032688")];

也许有人要问了,你这样做有什么好处呢?我导航到相应的控制器,我可以在需要导航的地方直接实例化且跳转过去就好了啊,干嘛要这么麻烦?

当某些功能页面有很多入口都可以进入的时候,我们可以采用这样的方式进行导航,避免了一个类的头文件被引用到多个类中,造成烂引用。从另一个角度来讲也保证了代码的整洁性。

当一个界面的的很多入口都是服务器动态控制的情况下,客户端可以省去根据相应的入口标识判断跳转到的页面的麻烦,也可以免去传参数的麻烦。只要服务端和客户端协定好自定义url的场景和规则,提前写好对应的实现,仅需一行代码,即可轻松搞定。从另一个角度来讲就是为ViewController瘦身。

上一篇 下一篇

猜你喜欢

热点阅读