看了前面的文章,相信很多同学还不知道RxSwift
该怎么使用,这篇文件将带领大家一起写一个 注册登录(ps:本例子采用MVVM
)的例子进行实战。本篇文章是基于RxSwift3.0
写的,采用的是Carthage
第三方管理工具导入的RxSwift3.0
,关于Carthage
的安装和使用,请参考Carthage的安装和使用。
下载Demo点我
首先请大家新建一个swift
工程,然后把RxSwift
引入到项目中,然后能够编译成功就行。
然后我们来分析下各个界面的需求:
好了,分析完上面的需求之后,是时候展示真正的技术了,let's go。
大家现在storyboard
中建立出下面这个样子的界面(ps:添加约束不在本篇范围内):
然后建立一个对应的控制器RegisterViewController
类,另外创建一个RegisterViewModel.swift
,将RegisterViewController
与storyboard
中的控制器关联,RegisterViewController
看起来应该是这样子的:
class RegisterViewController: UIViewController {
@IBOutlet weak var userNameTextField: UITextField!
@IBOutlet weak var nameLabel: UILabel!
@IBOutlet weak var pwdTextField: UITextField!
@IBOutlet weak var pwdLabel: UILabel!
@IBOutlet weak var rePwdTextField: UITextField!
@IBOutlet weak var rePwdLabel: UILabel!
@IBOutlet weak var registButton: UIButton!
@IBOutlet weak var loginButton: UIBarButtonItem!
override func viewDidLoad() {
super.viewDidLoad()
}
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
// Dispose of any resources that can be recreated.
}
}
另外,我们创建一个Service.swift
文件。Service
文件主要负责一些网络请求,和一些数据访问的操作。然后供ViewModel
使用,由于本次实战没有使用到网络,所以我们只是模拟从本地plist
文件中读取用户数据。
首先我们在Service
文件中创建一个ValidationService
类,最好不要继承NSObject
,Swift
中推荐尽量使用原生类。我们考虑到当文本框内容变化的时候,我们需要把文本框的内容当做参数传递进来进行处理,判断是否符合我们的要求,然后返回处理结果,也就是状态。基于此,我们创建一个Protocol.swift
文件,创建一个enum
用于表示我们处理结果,所以,我们在Protocol.swift
文件中添加如下代码:
enum Result {
case ok(message:String)
case empty
case failed(message:String)
}
先写出总结:其实就是两个流的传递过程。
UI操作 -> ViewModel -> 改变数据
数据改变 -> ViewModel -> UI刷新
回到我们Service
中ValidationService
类中,写一个检测username
的方法。它看起来应该是这个样子的:
class ValidationService {
// 单例类
static let instance = ValidationService()
private init(){}
let minCharactersCount = 6
func validationUserName(_ name:String) -> Observable {
if name.characters.count == 0 { // 当字符串为空的时候,什么也不做
return Observable.just(Result.empty)
}
if name.characters.count < minCharactersCount {
return Observable.just(Result.failed(message: "用户名长度至少为6位"))
}
if checkHasUserName(name) {
return Observable.just(Result.failed(message: "用户名已存在"))
}
return Observable.just(Result.ok(message: "用户名可用"))
}
func checkHasUserName(_ userName:String) -> Bool {
let filePath = NSHomeDirectory() + "/Documents/users.plist"
guard let userDict = NSDictionary(contentsOfFile: filePath) else {
return false
}
let usernameArray = userDict.allKeys as NSArray
return usernameArray.contains(userName)
}
}
接下来该处理我们的RegisterViewModel
了,我们声明一个username
,指定为Variable
类型,为什么是一个Variable
类型?因为它既是一个Observer
,又是一个Observable
,所以我们声明它是一个Variable
类型的对象。我们对username
处理应该会有一个结果,这个结果应该是由界面监听来改变界面显示,因此我们声明一个usernameUseable
表示对username
处理的一个结果,因为它是一个Observable
,所以我们将它声明为Observable
类型的对象,所以RegisterViewModel
看起来应该是这样子的:
class RegisterViewModel {
let username = Variable("")
let usernameUseable:Observable
init() {
}
}
然后我们再写RegisterViewController
,它看起来应该是这样子的:
private let disposeBag = DisposeBag()
override func viewDidLoad() {
super.viewDidLoad()
let viewModel = RegisterViewModel()
userNameTextField.rx.text.orEmpty.bind(to: viewModel.username).disposed(by: disposeBag)
}
userNameTextField.rx.text.orEmpty
是RxCocoa
库中的东西,它把TextFiled
的text
变成了一个Observable
,后面的orEmpty
我们可以Command
点进去看下,它会把String?
过滤nil
帮我们变为String
类型。 bind(to:viewModel.username)
的意思是viewModel.username
作为一个observer
(观察者)观察userNameTextField
上的内容变化。 disposeBag
来盛放我们这些监听的资源。 现在,回到我们的RegisterViewModel
中,我们添加如下代码:
init() {
let service = ValidationService.instance
usernameUseable = username.asObservable().flatMapLatest{ username in
return service.validationUserName(username).observeOn(MainScheduler.instance).catchErrorJustReturn(.failed(message: "userName检测出错")).shareReplay(1)
}
}
viewModel
中,我们把username
当做observable
(被观察者),然后对里面的元素进行处理之后发射对应的事件。 下面我们在RegisterViewController
中处理我们的username
请求结果。我们在ViewDidLoad
中添加下列代码:
viewModel.usernameUseable.bind(to:
nameLabel.rx.validationResult).addDisposableTo(disposeBag)
viewModel.usernameUseable.bind(to:
pwdTextField.rx.inputEnabled).addDisposableTo(disposeBag)
ViewModel
中username
处理结果usernameUseable
绑定到nameLabel
显示文案上,根据不同的结果显示不同的文案; ViewModel
中username
处理结果usernameUseable
绑定到pwdTextField
,根据不同的结果判断是否可以输入。 关于上面的validationResult
和inputEnabled
是需要我们自己去定制的,这就用到了RxSwift 系列(九) -- 那些难以理解的概念文章中的UIBindingObserver
了。
所以,我们在Protocol.swift
文件中添加如下代码:
extension Result {
var isValid:Bool {
switch self {
case .ok:
return true
default:
return false
}
}
}
extension Result {
var textColor:UIColor {
switch self {
case .ok:
return UIColor(red: 138.0 / 255.0, green: 221.0 / 255.0, blue: 109.0 / 255.0, alpha: 1.0)
case .empty:
return UIColor.black
case .failed:
return UIColor.red
}
}
}
extension Result {
var description: String {
switch self {
case let .ok(message):
return message
case .empty:
return ""
case let .failed(message):
return message
}
}
}
extension Reactive where Base: UILabel {
var validationResult: UIBindingObserver {
return UIBindingObserver(UIElement: base) { label, result in
label.textColor = result.textColor
label.text = result.description
}
}
}
extension Reactive where Base: UITextField {
var inputEnabled: UIBindingObserver {
return UIBindingObserver(UIElement: base) { textFiled, result in
textFiled.isEnabled = result.isValid
}
}
}
Result
进行了扩展,添加了isValid
属性,如果状态是ok
,这个属性就为true
,否则为false
Result
添加了一个textColor
属性,如果状态为ok
则为绿色,否则使用红色 UILabel
进行了UIBingObserver
,根据result
结果,进行它的text
和textColor
显示 UITextField
进行了UIBingObserver
,根据result
结果,对它的isEnabled
进行设置。 写到这里,我们暂停一下,运行一下项目看下程序的运行情况,试着去输入username
尝试一下效果,是不是很激动??
有了上面username
的理解,相信大家对password
也就熟门熟路了,因此有些细节就不做描述了。
我们现在对Service
中添加对password
的处理:
func validationPassword(_ password:String) -> Result {
if password.characters.count == 0 {
return Result.empty
}
if password.characters.count < minCharactersCount {
return .failed(message: "密码长度至少为6位")
}
return .ok(message: "密码可用")
}
func validationRePassword(_ password:String, _ rePassword: String) -> Result {
if rePassword.characters.count == 0 {
return .empty
}
if rePassword.characters.count < minCharactersCount {
return .failed(message: "密码长度至少为6位")
}
if rePassword == password {
return .ok(message: "密码可用")
}
return .failed(message: "两次密码不一样")
}
validationPassword
处理我们输入的密码; validationRePassword
处理我们输入的重复密码; Result
类型的值,因为我们外面不需要对这个过程进行监听,所以不必返回一个新的序列。 在RegisterViewModel
中添加需要的observable
:
let password = Variable("")
let rePassword = Variable("")
let passwordUseable:Observable
let rePasswordUseable:Observable
然后在init()
中初始化passwordUseable
和rePasswordUseable
:
passwordUseable = password.asObservable().map { passWord in
return service.validationPassword(passWord)
}.shareReplay(1)
rePasswordUseable = Observable.combineLatest(password.asObservable(), rePassword.asObservable()) {
return service.validationRePassword($0, $1)
}.shareReplay(1)
回到RegisterViewController
中,添加对应的绑定:
pwdTextField.rx.text.orEmpty.bind(to: viewModel.password).disposed(by: disposeBag)
rePwdTextField.rx.text.orEmpty.bind(to: viewModel.rePassword).disposed(by: disposeBag)
viewModel.passwordUseable.bind(to: pwdLabel.rx.validationResult).addDisposableTo(disposeBag)
viewModel.passwordUseable.bind(to: rePwdTextField.rx.inputEnabled).addDisposableTo(disposeBag)
viewModel.rePasswordUseable.bind(to: rePwdLabel.rx.validationResult).addDisposableTo(disposeBag)