(整个关于ReactiveCocoa的代码工程可以在https://github.com/qianhongqiang/QHQReactive下载)
好多天没写东西了,今天继续。主要讲解RAC如何于UI空间实现响应流的。
随手找个按钮响应的RAC实现作为示例,然后我们去做一个简单的实现
[[_HiddenBtn rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
_HiddenBtn.hidden = YES;
}];
我把一些别的逻辑全部删除,大家也不要关注任何内存问题。这个示例的功能很简单,就是一个按钮被点击抬起这个事件触发后,按钮隐藏。嗯,实现的非常优雅,逻辑也整合到一块去了,代码清晰明了。
我们实现的思路是这样的,创建一个信号,将这个信号与按钮关联,每次触发点击的时候,将值传递给这个信号,沿着信号管道去传递给订阅者。
-(QHQSignal *)qhq_signalForControlEvents:(UIControlEvents)event;我们添加一个UIControl分类,添加类似的方法。
然后考虑一下正常情况下按钮的响应通常是如何添加的,没错,很常见的
[按钮 addTarget:目标对象 action:目标对象方法 forControlEvents:响应事件];
来看看创建信号的方法
+(QHQSignal *)createSignal:(void(^)(id subscriber))didSubscriber
我们之前一直是让这个匿名内建的subscriber去发送消息,比如sendNext等事件,所以顺着这个思路,只需要把addTarget的目标添加为这个subscriber,调用subscriber的sendNext方法就可以了。然后我们来实现一下
-(QHQSignal *)qhq_signalForControlEvents:(UIControlEvents)event {
return [QHQSignal createSignal:^(id<QHQSubscrib> subscriber) {
[self addTarget:subscriber action:@selector(sendNext:) forControlEvents:event];
}];
}
是的,就这一句话就行了。然后来测试以下,我们创建个按钮,然后点击后输出些什么
[[demoButton qhq_signalForControlEvents:UIControlEventTouchDown] subscribeNext:^(id x) {
NSLog(@"%@---被点击了",x);
}];}
结果我点了半天,发现屏幕没有任何输出,怎么回事呢?然后我就开始调试,沿着整个栈信息找,没有什么问题,该创建的都创建了,可以是subscriber的sendNext方法不调用。我重写了一下subscriber的dealloc方法,插了一个断点,预料的一样,进断点了,也就是subscriber被干掉了。
我马上去翻了下API文档,展示一下
// passing in nil as the target goes up the responder chain. The action may optionally include the sender and the event in that order
// the action cannot be NULL. Note that the target is not retained.
- (void)addTarget:(nullable id)target action:(SEL)action forControlEvents:(UIControlEvents)controlEvents;
赫然写着Note that the target is not retained.后来想想,这也是必然的,不然按照常规写法,那不都跟VC循环引用了,自己小白了。
实际上,在RAC中对生命周期的管理做的还是很到位,这里暂时不展开,涉及到的东西比较多。以学习目的为主,我暂时将这个subscriber进行一次retain。
-(QHQSignal *)qhq_signalForControlEvents:(UIControlEvents)event {
return [QHQSignal createSignal:^(id<QHQSubscrib> subscriber) {
[self addTarget:subscriber action:@selector(sendNext:) forControlEvents:event];
objc_setAssociatedObject(self, _cmd, subscriber, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}];
}
这样代码可以正常运行了。
2016-01-07 13:50:43.142 PageText[10115:8236159] <UIButton: 0x7ff30a771730; frame = (0 200; 320 40); opaque = NO; layer = <CALayer: 0x7ff30a70c930>>---被点击了
在这处理中,自己也是长进不少。剩余的控件大家可以顺着这个思路自己摸索。