一、回顾
在上一篇中,我们着重介绍了如何将现有的网络库进行RAC封装。这篇将讲解具体的使用。先从ViewModelServices说起。
二、DDViewModelServices
目前,app业务层的整体架构如上图所示。在ViewModel和ViewModel之间传递了DDViewModelServicesImpl(实现了DDViewModelServices协议的)对象。如:
STTaskDetailViewModel *viewModel = [[STTaskDetailViewModel alloc] initWithServices:self.services params:@{}];
[self.services pushViewModel:viewModel animated:YES];
由上图可以看出,所有网络相关的调用都是在ViewModel中进行的,ViewModel负责数据的获取以及处理。那么我们给DDViewModelServices增加一个property。
@protocol DDViewModelServices <NSObject, DDNavigationProtocol>
@property (nonatomic, strong) DDNetWorkManger *client;
@end
然后在app打开之后,初始化首页的时候,给实现了DDViewModelServices协议的DDViewModelServicesImpl对象,初始化网络管理对象client。这样,在ViewModel和ViewModel之间push或者present的时候,就可以把对应的client进行传递了:
// 初始化client
DDNetWorkManger *client = [DDNetWorkManger defaultManger];
// 这个client通过services在ViewModel之间传递。
DDSharedAppDelegate.services.client = client;
OrderListViewModel *viewModel = [[OrderListViewModel alloc] initWithServices:DDSharedAppDelegate.services params:nil];
其中DDSharedAppDelegate是一个宏。
#define DDSharedAppDelegate ((AppDelegate *)([UIApplication sharedApplication].delegate))
AppDelegate保持了一个DDViewModelServicesImpl,所有后续ViewModel的services都是从这里传递的。
AppDelegate.h
@interface AppDelegate : DDAppDelegate
@property (nonatomic, strong, readonly) DDViewModelServicesImpl *services;
...
@end
AppDelegate.m
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
...
self.services = [[DDViewModelServicesImpl alloc] init];
...
return YES;
}
到这里,清楚了网络请求发生的层级,以及client对象是如何传递的。那么,我们下一步要结合DDNetWorkManger+RAC
来进行不同业务模块的网络请求信号封装。
三、DDNetWorkManger扩展
其中DDNetWorkManger+RAC
是网络库的RAC封装。DDNetWorkManger+SevenFresh
、DDNetWorkManger+TaskFlow
、DDNetWorkManger+MyTask
都是和具体业务有关的网络库扩展,分别为7-fresh、订单流程、我的任务业务模块。当然,随着业务的发展,可以一直增加对应的DDNetWorkManger
业务扩展。每个业务扩展里面只封装与该业务相关的网络层请求:
DDNetWorkManger+TaskFlow.h
@interface DDNetWorkManger (TaskFlow)
...
// 根据类型,获取任务列表。
- (RACSignal *)signal_requestTaskListWithType:(NSNumber *)type;
// 请求派单数据。
- (RACSignal *)signal_requestRecommendTaskWithIdentifier:(NSString *)taskID;
...
@end
DDNetWorkManger+TaskFlow.m
@implementation DDNetWorkManger (TaskFlow)
...
#pragma mark -
- (RACSignal *)signal_requestTaskListWithType:(NSNumber *)type
{
...
DDRequestInfo *request = [DDRequestInfo xxxxxx];
return [[self enqueueRequest:request method:DDHTTPMethodGET resultClass:[DDAPIResult class]] dd_parsedResults];
}
- (RACSignal *)signal_requestRecommendTaskWithIdentifier:(NSString *)taskID
{
...
DDRequestInfo *request = [DDRequestInfo xxxxx];
return [[self enqueueRequest:request method:DDHTTPMethodGET resultClass:[DDAPIResult class]] dd_parsedResults];
}
...
@end
到这里,我们就将不同业务对应的网络层请求,借助DDNetWorkManger+RAC
封装成了信号。
四、订阅网络层信号
我们给ViewModelServices增加了网络层DDNetWorkManger
类型的Client
。然后,将实现这个Services协议的DDViewModelServicesImpl类型的对象services在ViewModel之间传递。然后通过这个services.client订阅DDNetWorkManger
对应的业务扩展中的信号。
STTaskFlowViewModel.h
@interface STTaskFlowViewModel : DDTableViewModel
...
// 任务列表
@property (nonatomic, strong, readonly) RACCommand *requestTaskListCommand;
...
@end
STTaskFlowViewModel.m
- (RACCommand *)requestTaskListCommand
{
if (!_requestTaskListCommand)
{
WSELF;
_requestTaskListCommand = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(NSNumber *type) {
SSELF;
return [self.services.client signal_requestTaskListWithType:type];
}];
}
return _requestTaskListCommand;
}
然后在需要获取数据的地方如下调用即可:
[[self.requestTaskListCommand.executionSignals.switchToLatest map:^id _Nullable(id _Nullable value) {
...
return value;
}] subscribeNext:^(DDAPIResult *result) {
...
}];
五、总结
首先创建DDNetWorkManger
的相应业务扩展,并将DDNetWorkManger+RAC
包含进来。
然后在扩展中封装网络请求的信号。
再在ViewModel中实现对网络请求信号的订阅。