一般登录界面都会有一用户名和密码输入框,还有一个登录按钮,这次仅仅为了展现一个登录的,不涉及注册和忘记密码,记住密码此类功能。
RAC里面很多都是以block的形式出现的,这样会造成内存警告,所以务必要写上typeof(self)weakSelf = self;之后调用方法或者属性用weakSelf代替
考虑UE
1:假设现在要求 用户名长度 >=2且密码长度>=3,登录按钮可用。由于现在RAC是以Signal为机制的,那么用户名长度变化为一个Signal命名为validUsernameSignal密码长度变化为一个Signal 命名为validPasswordSignal,且在两者输入无效的情况下,输入框为黄色,反之白色 。即
//为了简单的监测用户输入的有效
-(BOOL) isValidInput:(NSString *)text tag:(NSInteger)tag{
BOOL flag = NO;
if (tag == 0) {
//用户名
flag = text.length >= 2;
}else{
//密码
flag = text.length >= 3;
}
return flag;
}
typeof(self)weakSelf = self;
RACSignal *validUsernameSignal = [_nameTextField.rac_textSignal map:^id(id value) {
return @([weakSelf isValidInput:value tag:0]);
}];
RACSignal *validPasswordSignal = [_pwdTextField.rac_textSignal map:^id(NSString *text) {
return @([weakSelf isValidInput:text tag:1]);
}];
//转换这些信号,从而能为输入框设置不同的背景颜色。
RAC(_nameTextField, backgroundColor) =
[validUsernameSignal
map:^id(NSNumber *nameValid){
return[nameValid boolValue] ? [UIColor clearColor]:[UIColor yellowColor];
}];
RAC(_pwdTextField,backgroundColor) = [validPasswordSignal
map:^id(NSNumber *passwordValid){
return[passwordValid boolValue] ? [UIColor clearColor]:[UIColor yellowColor];
}];
2: 把以上两个信号组合一起,才能真正的表示登录按钮的可用性(enabled)
//信号聚合 使用combineLatest:reduce:方法把validUsernameSignal和validPasswordSignal产生的最新的值聚合在一起,并生成一个新的信号。每次这两个源信号的任何一个产生新值时,reduce block都会执行,block的返回值会发给下一个信号。
RACSignal *signUpActiveSignal = [RACSignal combineLatest:@[validUsernameSignal,validPasswordSignal] reduce:^id(NSNumber*usernameValid, NSNumber *passwordValid){
return @([usernameValid boolValue]&&[passwordValid boolValue]);
}];
[signUpActiveSignal subscribeNext:^(NSNumber*signupActive){
_loginBtn.enabled = [signupActive boolValue];
_loginBtn.alpha = [signupActive boolValue] ? 1:0.4;
}];
3: 当登录按钮可用,点击登录按钮,如果登录成功,则进入登录成功界面反之失败界面。这里假设用户名输入的为“123”则登录成功。
[[[_loginBtn rac_signalForControlEvents:UIControlEventTouchUpInside]flattenMap:^RACStream *(id value) {
return [weakSelf signInSignal];
}]subscribeNext:^(id x) {
[weakSelf IsloginSuccess:[x boolValue]];
}];
- (RACSignal *)signInSignal{
[self resignResponder];
typeof(self)weakSelf = self;
return [RACSignal createSignal:^RACDisposable *(id subscriber){
[weakSelf login:_nameTextField.text password:_pwdTextField.text block:^(BOOL obj) {
[subscriber sendNext:@(obj)];
[subscriber sendCompleted];
}];
return nil;
}];
}
-(void)IsloginSuccess:(BOOL)flag{
SuccessViewController *uisvc = [[ SuccessViewController alloc]init];
[self.navigationController pushViewController:uisvc animated:true];
uisvc.title = flag ?@"登录成功": @"登录失败";
}
- (void) login:(NSString * )carNumber password:(NSString *)pwd block:(Block_Bool)result{
//Block_Bool的define定义 :typedef void (^Block_Bool) (BOOL b);
if ([_nameTextField.text isEqualToString:@"123"]) {
result(YES);
return;
}
result(NO);
}
4:为了更好的展示UE当点击self.view时,让键盘隐藏,这时加个手势:
UITapGestureRecognizer *tap = [[UITapGestureRecognizer alloc]init];
tap.numberOfTapsRequired = 1;
tap.numberOfTouchesRequired = 1;
[self.view addGestureRecognizer:tap];
// 这是给self.view加上一个事件,点击的时候让键盘隐藏。
[tap rac_signalForSelector:@selector(resignResponder)];
RACSignal *tapSignal = [tap rac_gestureSignal];
[[tapSignal map:^id(id value) {
return value;
}] subscribeNext:^(id x) {
[weakSelf.view endEditing:YES];
}];