ReactiveCocoa开发中常见用法。
代替代理:
rac_signalForSelector:用于替代代理。
代替KVO :
rac_valuesAndChangesForKeyPath:用于监听某个对象的属性改变。
监听事件:
rac_signalForControlEvents:用于监听某个事件。
代替通知:
rac_addObserverForName:用于监听某个通知。
监听文本框文字改变:
rac_textSignal:只要文本框发出改变就会发出这个信号。
处理当界面有多次请求时,需要都获取到数据时,才能展示界面
rac_liftSelector:withSignalsFromArray:Signals:当传入的Signals(信号数组),每一个signal都至少sendNext过一次,就会去触发第一个selector参数的方法。
使用注意:几个信号,参数一的方法就几个参数,每个参数对应信号发出的数据。
温馨提示:Demo用到的第三方
导入的第三方框架
pod 'ReactiveCocoa', '~> 2.5'
pod 'Masonry', '~> 0.6.4'
pod 'LxDBAnything', '1.1.0'
代码:
@weakify(Obj)和@strongify(Obj),一般两个都是配套使用,在主头文件(ReactiveCocoa.h)中并没有导入,需要自己手动导入,RACEXTScope.h才可以使用。但是每次导入都非常麻烦,只需要在主头文件自己导入就好了。
/*************************************基础部分**************************************************/
#pragma mark --textField属性变化
-(void)textFieldChange
{
//初始化一个textField控件
UITextField *textField = ({
UITextField *textField = [[UITextField alloc] init];
textField.backgroundColor = [UIColor cyanColor];
textField ;
});
textField.delegate = self ;
[self.view addSubview:textField];
@weakify(self);
[textField mas_makeConstraints:^(MASConstraintMaker *make) {
@strongify(self);
make.size.mas_equalTo(CGSizeMake(180, 40));
make.center.equalTo(self.view);
}];
/*****************监听textField的属性变化情况*******************/
//RAC内部封装好的类
//默认执行1次,所以会有打印,原因:请查看RAC底层bind,hook的思想
//监听文本框的文字改变
[textField.rac_textSignal subscribeNext:^(id x) {
LxDBAnyVar(x);
}];
}
#pragma mark -- Button用法 监听事件
-(void)textBtn
{
UIButton *button = [UIButton buttonWithType:UIButtonTypeSystem];
[button setTitle:@"点击" forState:UIControlStateNormal];
button.backgroundColor = [UIColor redColor];
[self.view addSubview:button];
@weakify(self);
[button mas_makeConstraints:^(MASConstraintMaker *make) {
@strongify(self);
make.size.mas_equalTo(CGSizeMake(self.view.bounds.size.width, 30));
make.bottom.equalTo(self.view).offset(0);
}];
// 监听事件
// 把按钮点击事件转换为信号,点击按钮,就会发送信号
[[button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
LxPrintf(@"进来了");
RACViewController *VC = [[RACViewController alloc] init];
@strongify(self);
[self presentViewController:VC animated:YES completion:^{
LxPrintf(@"%s",__func__);
}];
}];
}
#pragma mark --手势
-(void)addTap
{
/**************手势*****************/
self.view.userInteractionEnabled = YES ;
UITapGestureRecognizer *tap = [[UITapGestureRecognizer alloc] init];
[[tap rac_gestureSignal] subscribeNext:^(id x) {
LxDBAnyVar(x);
}];
[self.view addGestureRecognizer:tap];
}
#pragma mark --通知中心
-(void)notifiCenter
{
/**************通知*****************/
[[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIApplicationDidEnterBackgroundNotification object:nil] subscribeNext:^(id x) {
LxDBAnyVar(x);
}];
}
#pragma mark --代理
-(void)delegateDemo
{
UIAlertView *alterView = [[UIAlertView alloc] initWithTitle:@"RAC" message:@"ReactiveCocoa" delegate:self cancelButtonTitle:@"Cancel" otherButtonTitles:@"Ensure", nil];
//方法一:代理
// [[self rac_signalForSelector:@selector(alertView:clickedButtonAtIndex:) fromProtocol:@protocol(UIAlertViewDelegate)] subscribeNext:^(id x) {
//
// RACTuple *tuple = (RACTuple *)x ; //类似于swift的元组
//
// LxDBAnyVar(tuple);
//
// LxDBAnyVar(tuple.first);
// LxDBAnyVar(tuple.second);
// LxDBAnyVar(tuple.third);
//
// }];
[alterView show];
//方法二:内部封装的方法
//更简单的方式
[[alterView rac_buttonClickedSignal] subscribeNext:^(id x) {
LxDBAnyVar(x);
}];
}
#pragma mark -- KVO
-(void)addKvo
{
UIScrollView *scrollView = [[UIScrollView alloc] init];
scrollView.delegate = (id<UIScrollViewDelegate>)self ;
[self.view addSubview:scrollView];
UIView *scrollViewContentView = [[UIView alloc] init];
scrollViewContentView.backgroundColor = [UIColor yellowColor];
[scrollView addSubview:scrollViewContentView];
@weakify(self);
[scrollView mas_makeConstraints:^(MASConstraintMaker *make) {
@strongify(self);
make.edges.equalTo(self.view).insets(UIEdgeInsetsMake(80, 80, 80, 80));
}];
[scrollViewContentView mas_makeConstraints:^(MASConstraintMaker *make) {
@strongify(self);
make.edges.equalTo(scrollView);
make.size.mas_equalTo(CGRectGetWidth(self.view.frame), CGRectGetHeight(self.view.frame));
}];
[RACObserve(scrollView, contentOffset) subscribeNext:^(id x) {
LxDBAnyVar(x);
}];
}
//处理当界面有多次请求时,需要都获取到数据时,才能展示界面
-(void)moreRequest
{
/*
1.rac_liftSelector:withSignalsFromArray:Signals:当传入的Signals(信号数组),每一个signal都至少sendNext过一次,就会去触发第一个selector参数的方法。
2.使用注意:几个信号,参数一的方法就几个参数,每个参数对应信号发出的数据。
*/
//处理多个请求,都返回结果的时候,统一做处理.
RACSignal *request1 = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
// 发送请求1
[subscriber sendNext:@"发送请求1"];
return nil;
}];
RACSignal *request2 = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
// 发送请求2
[subscriber sendNext:@"发送请求2"];
return nil;
}];
// 使用注意:几个信号,参数一的方法就几个参数,每个参数对应信号发出的数据。
[self rac_liftSelector:@selector(updateUIWithR1:r2:) withSignalsFromArray:@[request1,request2]];
}
// 更新UI
- (void)updateUIWithR1:(id)data r2:(id)data1
{
NSLog(@"更新UI%@ %@",data,data1);
}
#pragma mark --定时器
-(void)RACSchedulerAndMain
{
/**************定时器*****************/
//1.延迟某个时间再做某件事
// [[RACScheduler mainThreadScheduler] afterDelay:2 schedule:^{
//
// LxDBAnyVar(rac);
//
// }];
//2.每间隔多长时间做一件
[[RACSignal interval:1 onScheduler:[RACScheduler mainThreadScheduler]] subscribeNext:^(id x) {
LxDBAnyVar(x);
// LxPrintAnything(@"55555555");
}];
//这是定时器最常用的两种方法,第一种方法,延迟时间去做某件事,更改afterDelay的属性,第二种方法,每间隔躲藏时间去做一件事,更改interval的属性
}