- 背景
大型app的开发过程,伴随着多个业务线的同时进行。上层业务之间不允许出现耦合,但是免不了业务页面之间会相互跳转。因此,引入间接的导航器,通过它,将页面之间的依赖下沉,解除上层业务之间的耦合。
但是这样,导航器注入了所有的依赖,在OC中,可以通过反射,用字符串匹配类,来解除这些依赖。可是对于Swift,由于个人水平有限,还没有找到办法解决这个问题。如果你有好的解决方案,谢谢你告诉我。
二 实现
虽然Swift版导航器,背离了解耦的初衷,但是它仍然是有存在意义的。导航器的用法,通过url来进行跳转,可以避免直接依赖,免于写重复代码。像下面这样,一句话能跳转到对应的页面:
Navigator.open(“hello”)
Swift动态能力的缺乏,使得通过url映射某个类,必须是显示的把url和类进行注册。于是有了下面这个方法,通过字典将url和类型保存起来。
func register(_ host: String, _ type: Any) {
urlMap[host] = type
}
在程序启动时,调用register方法集中注册:
Navigator.register("hello", MyController.self)
Navigator.register("world", YourController.self)
一般的做法,在调用导航器跳转方法时,拿到类型后,会生成对应的页面实例,根据参数push或者present。但是这样,对页面的约束较大,必须拥有统一的初始化方法,固定的参数等,而且这个导航器的作用就只能用来进行页面跳转了。我们换个思路,拿到类型后,要做的事,交给类自己,导航器只做一层类方法转发调用,这样就很灵活。于是,设计NBNavigatable协议,它的唯一方法是类被导航后要做的事,所需要的参数都从NBURL(NBURL是一个封装的url对象)里取得,实现中你可以做任何事,push到界面上或者弹一个提示框,都可以:
protocol NBNavigatable {
static func opened(_ url: NBURL) -> Any?
}
能够被导航的类,需要遵从这个协议:
extension MyController : NBNavigatable {
static func opened(_ url: NBURL) -> Any? {
let controller = MyController.self();
return controller
}
}
有了协议后,注册方法需要做如下改动,只能注册满足协议的类型,而且保存类型的opened方法即可。
func register(_ host: String, _ type: NBNavigatable.Type) {
urlMap[host] = type.opened
}
跳转方法,参数变成NBURL,将需要传递的参数都塞进去,被跳转类可以通过 __func object(ForKey key: String) -> Any? __取到对应的值。
let url = NBURL("myapp://hello?title=world")
url.setObject("13100001111", forKey: "number")
Navigator.open(url)
这里的scheme需要在程序启动时设置的,有效的scheme才使用导航器跳转,外部的非法的scheme使用系统方法进行跳转。
当然,导航器大部分作用还是页面跳转,于是写了一个UIViewController的扩展,封装了常见的页面跳转实现。
三 用法
总结一下使用过程:
1、程序启动时,注册url与类,并且设置有效scheme;
2、可以被跳转的类遵从NBNavigatable协议,在opened方法里实现被跳转时要做的事;
3、使用open方法跳转。
源码见这里 ,欢迎给我提意见。
网友评论