URLNavigator是Swift版本的Router。
Router的主要作用是解耦。之前在各个ViewController
间跳转,需要import ViewController
,这样就造成ViewController
之间的依赖,也即耦合。通过router
不需要再import ViewController
。所有的只要import router
,只依赖router
这一个类,router
里再去import
其他的ViewController
,这样,就达到我们说的解耦。
一个基本完善的router
,我认为应该有下面几个核心功能:
- 跳转
ViewController
- 跳转服务
- 回传值
跳转ViewController是基本功能,这里包括跳转的时候,传入参数。
对于跳转,我们需要做到通过一个字符串,来跳转到我们想要的页面,那么我们首先要做的是将字符串和对应的页面关联起来,到时候,你给我这个字符串,我就知道你需要去哪个页面。
在URLNavigator
里有一个注册方法,就是将字符串和需要跳转的ViewController
关联起来。
navigator.register("navigator://user") { url, values, context in
return UserViewController()
}
更进一步,可以在字符串里把需要传递的参数也带上
navigator.register("navigator://user/<username>") { url, values, context in
guard let username = values["username"] as? String else { return nil }
return UserViewController(navigator: navigator, username: username)
}
这样不仅可以跳转到关联页面,还能传递参数。
register
里面做的事很简单,就是用一个字典将字符串和和ViewController
关联起来。
public typealias URLPattern = String
private var viewControllerFactories = [URLPattern: ViewControllerFactory]()
open func register(_ pattern: URLPattern, _ factory: @escaping ViewControllerFactory) {
self.viewControllerFactories[pattern] = factory
}
将字符串作为字典的key
,创建ViewController
的闭包作为value
,就这样关联了字符串和ViewController
。
在调用的时候,再根据字符串找到相应的闭包,得到ViewController
,执行跳转动作。
open func viewController(for url: URLConvertible, context: Any? = nil) -> UIViewController? {
let urlPatterns = Array(self.viewControllerFactories.keys)
guard let match = self.matcher.match(url, from: urlPatterns) else { return nil }
guard let factory = self.viewControllerFactories[match.pattern] else { return nil }
return factory(url, match.values, context)
}
从viewControllerFactories
字典里拿到factory
,再执行factory(url, match.values, context)
。
这里的guard let match = self.matcher.match(url, from: urlPatterns) else { return nil }
是拿到url里的参数,这里面参数的传入有一个自己定义的规则。URLMatcher.swift
就是专门处理字符串的拆分,拿到参数。
然后
navigator.push("navigator://user/zhangsan")
navigator.present("navigator://user/zhangsan")
调用服务,有时候,我们并不想跳到一个页面,仅仅是想调用某个类里面的某个函数。
navigator.register("navigator://user/<username>") { url, values, context in
guard let username = values["username"] as? String else { return nil }
return UserViewController(navigator: navigator, username: username)
}
这个是上面的注册代码,我们只要修改一下就可以了,调用服务,其他的操作一样
navigator.register("navigator://user/<username>") { url, values, context in
guard let username = values["username"] as? String else { return nil }
//获取UserViewController对象 userVC,调用方法
userVC.callFuc(username: username)
}
这个是我假想的一个方法。实际有一个和register类似的方法
private var handlerFactories = [URLPattern: URLOpenHandlerFactory]()
open func handle(_ pattern: URLPattern, _ factory: @escaping URLOpenHandlerFactory) {
self.handlerFactories[pattern] = factory
}
有一个保存闭包和字符串对应关系的字典handlerFactories
,不和register
字典共用。
所以调用方法是
navigator.handle("navigator://user/<username>") { (url, values, context) -> Bool in
guard let username = values["username"] as? String else { return nil }
//获取UserViewController对象 userVC,调用方法
userVC.callFuc(username: username)
return true
}
回传值,有时候,我们跳转到某个页面,需要这个页面执行后,把相关结果返回。
目前,URLNavigator
还没有第三个功能。
总结一下:自己定义一个字符串规则,包含页面信息和参数信息,然后将字符串和对应的闭包关联起来,闭包可以是创建相应ViewController
的操作,也可以是调用函数的操作,也可以是其他操作。在router
通过字符串跳转的时候,拿到字符串,解析出参数,找到相应的闭包,将参数传给闭包执行,执行闭包得到的ViewController
,拿去跳转。