上一篇我們已經(jīng)對RACSignal的大部分用法都概括了一遍,我們把它理解為一個(gè)信號(hào),可以比較清晰的在腦海里想象出來,但是RACCommand相對就要難上一些了!
RACCommand創(chuàng)建
有兩個(gè)方法可以創(chuàng)建,而且一般使用懶加載:
第一種
[RACCommand alloc]initWithSignalBlock:<#^RACSignal *(id input)signalBlock#>
第二種,帶開關(guān)信號(hào)
[RACCommand alloc]initWithEnabled:<#(RACSignal *)#> signalBlock:<#^RACSignal *(id input)signalBlock#>
很顯然,RACCommand的創(chuàng)建是離不開RACSignal的,如果對RACSignal都不了解,也就不用想運(yùn)用起RACCommand來了.首先,我們看那個(gè)開關(guān)信號(hào),它是個(gè)信號(hào),值是@YES或@NO,YES才可以執(zhí)行.后面的參數(shù)是個(gè)block,返回值也是個(gè)信號(hào),參數(shù)input是執(zhí)行命令源傳過來的值,比如后面要講的例子就是一個(gè)按鈕.
執(zhí)行RACCommand
上文中提到的input值是不是還是很迷惑呢,我們看看執(zhí)行命令的方法就會(huì)了解了:
[viewModel.command execute:@"(id)input"];
為了直觀我直接就給input傳了個(gè)字符串@"(id)input",然后你在命令初始化的地方就能打印到這個(gè)值了.
命令執(zhí)行的時(shí)候,信號(hào)必須[subscriber sendCompleted],否則命令就一直處于執(zhí)行狀態(tài).我們可以監(jiān)聽執(zhí)行狀態(tài):
[viewModel.command.executing subscribeNext:^(id x) {
if ([x boolValue]) {
NSLog(@"正在執(zhí)行");
}else{
NSLog(@"沒執(zhí)行/執(zhí)行完畢");
}
}];
理解(處理)命令
命令發(fā)出執(zhí)行起來后,我們需要接收到命令中包含的信號(hào),并處理之:
[viewModel.command.executionSignals subscribeNext:^(id x) {
// NSLog(@"===%@",x);//發(fā)現(xiàn)x是一個(gè)RACSignal(RACDynamicSignal)
// }];
executionSignals是個(gè)信號(hào)集,可能是完成信號(hào),可能是錯(cuò)誤信號(hào),而錯(cuò)誤信號(hào)需要另外接收:
[[viewModel.command.errors subscribeOn:[RACScheduler mainThreadScheduler]]map:^id(id value) {
return @"failed";
}];
這么多信號(hào),我們大部分時(shí)候需要的是最新的,拿到最新信號(hào)的方法:
[viewModel.command.executionSignals.switchToLatest subscribeNext:^(id x) {
}error:^(NSError *error) {
}completed:^{
}];
舉例說明
通過上面的概述我們能夠創(chuàng)建一個(gè)RACCommand,并且執(zhí)行起來,然后接收發(fā)出的信號(hào),處理傳過來的數(shù)據(jù),但是實(shí)際上運(yùn)用時(shí)還是有很多的彎彎繞不出來,下面通過一個(gè)實(shí)際的,典型的例子加深和梳理一下理解:
首先我們建個(gè)ViewModel類并且定義屬性:
@property (nonatomic,copy) NSString *text1;
@property (nonatomic,copy) NSString *text2;
@property (nonatomic,copy) NSString *text3;
@property(nonatomic, strong) RACSignal *enableSignal;
@property(nonatomic, strong) RACCommand *btnCommand;
RAC對按鈕有個(gè)特別偏心的地方就是直接給他創(chuàng)建了一個(gè)RACCommand,并且當(dāng)你點(diǎn)擊時(shí)自動(dòng)執(zhí)行,甚至把RACCommand的Enabled信號(hào)和按鈕的enabled屬性綁定起來,難道UIButton才是RACCommand的靈感來源?
言歸正傳,我們需要在.m中使用懶加載創(chuàng)建RACCommand:
//RACCommand的懶加載
- (RACCommand *)btnCommand{
if (!_btnCommand) {
_btnCommand = [[RACCommand alloc]initWithEnabled:self.enableSignal signalBlock:^RACSignal *(id input) {
NSLog(@"input = %@",input);
return [ViewModel doSomething];
}];
}
return _btnCommand;
}
+ (RACSignal *)doSomething{
return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
[subscriber sendNext:@"點(diǎn)擊按鈕總得干點(diǎn)什么"];
NSError *e = [[NSError alloc]initWithDomain:@"domain" code:110 userInfo:nil];
[subscriber sendError:e];
[subscriber sendCompleted];
return nil;
}];
}
然后我們在ViewModel的init方法中實(shí)現(xiàn)訂閱他的信號(hào):
- (void)setupViewModel{
// [self.btnCommand.executionSignals subscribeNext:^(id x) {
// NSLog(@"===%@",x);//發(fā)現(xiàn)x是一個(gè)RACSignal(RACDynamicSignal)
// }];
//1.實(shí)際運(yùn)用中這個(gè)可以用來強(qiáng)制傳值
//轉(zhuǎn)為字符串信號(hào)
RACSignal *strSignal = [self.btnCommand.executionSignals map:^id(id value) {
return @"我是一個(gè)RACSignal變身而來的字符串";
}];
RAC(self,text1) = strSignal;
//2.實(shí)際運(yùn)用中這個(gè)可以用來success,如果你監(jiān)聽RACEventTypeNext,就能得到參數(shù),但是不能的到error
RACSignal *strSignal2 = [self.btnCommand.executionSignals flattenMap:^RACStream *(id value) {
// NSLog(@"xx=%@",value);//value是一個(gè)新的RACSignal(RACDynamicSignal)
RACSignal *s = value;
RACSignal *es = [s materialize];//轉(zhuǎn)化為一個(gè)傳RACEvent的信號(hào)
//得到事件的狀態(tài)和值
RACSignal *fs = [[es filter:^BOOL(id value) {
RACEvent *ev = value;
if (ev.eventType == RACEventTypeNext) {
NSLog(@"next == %@",ev.value);
}
return @(ev.eventType == RACEventTypeCompleted);//Completed
}]map:^id(id value) {
return @"success";
}];
return fs;
}];
RAC(self,text2) = strSignal2;
//3.實(shí)際運(yùn)用中這個(gè)可以用來fail,2不能得到的這個(gè)可以得到了
RACSignal *strSignal3 = [[self.btnCommand.errors subscribeOn:[RACScheduler mainThreadScheduler]]map:^id(id value) {
return @"failed";
}];
RAC(self,text3) = strSignal3;
//這些信號(hào)也可以合并(merge)在一起,畢竟是一個(gè)command來的
}
這樣的話你在ViewController中就很簡單
- (void)btnCommand{
self.btn_1.rac_command = self.viewModel.btnCommand;
RAC(self.lbl_1,text) = RACObserve(self.viewModel, text1);
RAC(self.lbl_2,text) = RACObserve(self.viewModel, text2);
RAC(self.lbl_3,text) = RACObserve(self.viewModel, text3);
}
- (void)test{
self.viewModel = [ViewModel new];
[self btnCommand];
}
這樣你就實(shí)現(xiàn)了,點(diǎn)擊按鈕,同時(shí)改變View中三個(gè)label的顯示.demo后續(xù)上傳.