函数响应式编程及ReactiveObjC学习笔记 (四)
今天我们继续看其他的类别
UIImagePickerController+RACSignalSupport.h
#import <UIKit/UIKit.h> @class RACDelegateProxy;
@class RACSignal<__covariant ValueType>; NS_ASSUME_NONNULL_BEGIN @interface UIImagePickerController (RACSignalSupport) @property (nonatomic, strong, readonly) RACDelegateProxy *rac_delegateProxy; - (RACSignal<NSDictionary *> *)rac_imageSelectedSignal; @end NS_ASSUME_NONNULL_END
可以看到它有一个代理属性, 一个信号
再看下它有帮我们代理哪些方法
- (RACSignal *)rac_imageSelectedSignal {
RACSignal *pickerCancelledSignal = [[self.rac_delegateProxy
signalForSelector:@selector(imagePickerControllerDidCancel:)]
merge:self.rac_willDeallocSignal];
RACSignal *imagePickerSignal = [[[[self.rac_delegateProxy
signalForSelector:@selector(imagePickerController:didFinishPickingMediaWithInfo:)]
reduceEach:^(UIImagePickerController *pickerController, NSDictionary *userInfo) {
return userInfo;
}]
takeUntil:pickerCancelledSignal]
setNameWithFormat:@"%@ -rac_imageSelectedSignal", RACDescription(self)];
RACUseDelegateProxy(self);
return imagePickerSignal;
}
一个取消按钮的代理, 一个选择完毕的代理,
我们用来试试看
// 创建一个RACDelegateProxy
RACDelegateProxy *imgPickerDelegateProxy = [[RACDelegateProxy alloc] initWithProtocol:@protocol(UIImagePickerControllerDelegate)]; // 绑定取消代理事件
[[imgPickerDelegateProxy
rac_signalForSelector:@selector(imagePickerControllerDidCancel:)]
subscribeNext:^(RACTuple * _Nullable x) { NSLog(@"取消代理: %@", x);
UIImagePickerController *imgPicker = x[];
[imgPicker dismissViewControllerAnimated:YES completion:nil];
}]; // 绑定选择完毕事件
[[imgPickerDelegateProxy
rac_signalForSelector:@selector(imagePickerController:didFinishPickingImage:editingInfo:)]
subscribeNext:^(RACTuple * _Nullable x) { NSLog(@"选择完成: %@", x);
self.view.backgroundColor = [UIColor colorWithPatternImage:x[]]; UIImagePickerController *imgPicker = x[];
[imgPicker dismissViewControllerAnimated:YES completion:nil];
}]; // 创建UIImagePickerController
UIImagePickerController *imgPickerController = [[UIImagePickerController alloc] init];
imgPickerController.sourceType = UIImagePickerControllerSourceTypePhotoLibrary;
imgPickerController.allowsEditing = YES; // 设置代理为我们创建的delegateProxy
// 这里注意一般UIImagePickerController需要实现UIImagePickerDelegate & UINavigationControllerDelegate, 但这里用RAC自带的方法只能设置一个代理
// 不过其实RAC其实也只为UIImagePickerController处理了UIImagePickerControllerDelegate而已, 所以这里会有警告先不管它
// 后面我们再想想其他方式, 是重新写一个UIImagePickerController分类还是重写方法什么的 imgPickerController.delegate = (id<UIImagePickerControllerDelegate>)imgPickerDelegateProxy; // retain delegateProxy
objc_setAssociatedObject(imgPickerController, _cmd, imgPickerDelegateProxy, OBJC_ASSOCIATION_RETAIN_NONATOMIC); [self presentViewController:imgPickerController animated:YES completion:nil];
这里大家需要注意下红色字体部分,
点击取消返回的x是imgPickerController本身, 转义后直接拿来dismiss即可
选择图片后x是一个数组, 第一个是imgPickerComtroller, 第二个是选择的image
UISegmentedControl+RACSignalSupport.h
#import <UIKit/UIKit.h> @class RACChannelTerminal<ValueType>; NS_ASSUME_NONNULL_BEGIN @interface UISegmentedControl (RACSignalSupport) - (RACChannelTerminal<NSNumber *> *)rac_newSelectedSegmentIndexChannelWithNilValue:(nullable NSNumber *)nilValue; @end NS_ASSUME_NONNULL_END
可以把分段选择器的选择结果直接绑定给其他空间, 或者拿去做别的 实例代码如下:
// 创建一个UISegementController
UISegmentedControl *segmentController = [[UISegmentedControl alloc] initWithItems:@[@"One", @"Two"]];
segmentController.frame = CGRectMake(, , , );
segmentController.center = self.view.center; [self.view addSubview:segmentController]; // 创建一个UITextField
UITextField *textField = [[UITextField alloc] initWithFrame:CGRectMake(, , , )];
textField.center = CGPointMake(self.view.center.x, self.view.center.y + );
textField.layer.borderWidth = ;
textField.layer.borderColor = [UIColor blueColor].CGColor; [self.view addSubview:textField]; // 获取segement的channelTerminal
RACChannelTerminal *segmentTerminal = [segmentController rac_newSelectedSegmentIndexChannelWithNilValue:]; // 获取textField的channelTerminal
RACChannelTerminal *textFieldTerminal = [textField rac_newTextChannel]; // 把segementdeterminal结果处理后绑定给textfieldTerminal
[[segmentTerminal
map:^id _Nullable(id _Nullable value) { if ([value isEqual: @()]) { return @"选择了: Two";
} else { return @"选择了: One";
}
}]
subscribe:textFieldTerminal];
UISlider+RACSignalSupport.h
#import <UIKit/UIKit.h> @class RACChannelTerminal<ValueType>; NS_ASSUME_NONNULL_BEGIN @interface UISlider (RACSignalSupport) - (RACChannelTerminal<NSNumber *> *)rac_newValueChannelWithNilValue:(nullable NSNumber *)nilValue; @end NS_ASSUME_NONNULL_END
跟UISegement一样, 我们直接给上代码:
// 创建一个UISlider
UISlider *slider = [[UISlider alloc] initWithFrame:CGRectMake(, , , )];
slider.center = self.view.center; [self.view addSubview:slider]; // 创建一个UITextField
UITextField *textField = [[UITextField alloc] initWithFrame:CGRectMake(, , , )];
textField.center = CGPointMake(self.view.center.x, self.view.center.y + );
textField.layer.borderWidth = ;
textField.layer.borderColor = [UIColor blueColor].CGColor; [self.view addSubview:textField]; // 获取slider的channelTerminal
RACChannelTerminal *sliderTerminal = [slider rac_newValueChannelWithNilValue:]; // 获取textField的channelTerminal
RACChannelTerminal *textFieldTerminal = [textField rac_newTextChannel]; [[sliderTerminal
map:^id _Nullable(id _Nullable value) { return [NSString stringWithFormat:@"%@", value];
}]
subscribe:textFieldTerminal];
运行截图:

UIStepper+RACSignalSupport.h
#import <UIKit/UIKit.h> @class RACChannelTerminal<ValueType>; NS_ASSUME_NONNULL_BEGIN @interface UIStepper (RACSignalSupport) - (RACChannelTerminal<NSNumber *> *)rac_newValueChannelWithNilValue:(nullable NSNumber *)nilValue; @end NS_ASSUME_NONNULL_END
也是只有一个RACChannelTerminal, 我们直接上代码:
// 创建一个UIStepper
UIStepper *stepper = [[UIStepper alloc] initWithFrame:CGRectMake(, , , )];
stepper.center = self.view.center; [self.view addSubview:stepper]; // 创建一个UITextField
UITextField *textField = [[UITextField alloc] initWithFrame:CGRectMake(, , , )];
textField.center = CGPointMake(self.view.center.x, self.view.center.y + );
textField.layer.borderWidth = ;
textField.layer.borderColor = [UIColor blueColor].CGColor; [self.view addSubview:textField]; // 获取stepper的channelTerminal
RACChannelTerminal *stepperTerminal = [stepper rac_newValueChannelWithNilValue:]; // 获取textField的channelTerminal
RACChannelTerminal *textFieldTerminal = [textField rac_newTextChannel]; [[stepperTerminal
map:^id _Nullable(id _Nullable value) { return [NSString stringWithFormat:@"%@", value];
}]
subscribe:textFieldTerminal];
运行截图:

UISwitch+RACSignalSupport.h
#import <UIKit/UIKit.h> @class RACChannelTerminal<ValueType>; NS_ASSUME_NONNULL_BEGIN @interface UISwitch (RACSignalSupport) - (RACChannelTerminal<NSNumber *> *)rac_newOnChannel; @end NS_ASSUME_NONNULL_END
示例:
// 创建一个UISwitch
UISwitch *aswitch = [[UISwitch alloc] initWithFrame:CGRectMake(, , , )];
aswitch.center = self.view.center; [self.view addSubview:aswitch]; // 创建一个UITextField
UITextField *textField = [[UITextField alloc] initWithFrame:CGRectMake(, , , )];
textField.center = CGPointMake(self.view.center.x, self.view.center.y + );
textField.layer.borderWidth = ;
textField.layer.borderColor = [UIColor blueColor].CGColor; [self.view addSubview:textField]; // 获取aswitch的channelTerminal
RACChannelTerminal *aswitchTerminal = [aswitch rac_newOnChannel]; // 获取textField的channelTerminal
RACChannelTerminal *textFieldTerminal = [textField rac_newTextChannel]; [[aswitchTerminal
map:^id _Nullable(id _Nullable value) { if ([value boolValue]) { return @"";
} return @"";
}]
subscribe:textFieldTerminal];
UITableViewCell+RACSignalSupport.h
#import <UIKit/UIKit.h> @class RACSignal<__covariant ValueType>;
@class RACUnit; NS_ASSUME_NONNULL_BEGIN @interface UITableViewCell (RACSignalSupport) @property (nonatomic, strong, readonly) RACSignal<RACUnit *> *rac_prepareForReuseSignal; @end NS_ASSUME_NONNULL_END
有一个rac_prepareForReuseSignal属性, 看字面意思就很清除准备复用时调用
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"Cell"];
if (!cell) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:@"Cell"];
}
cell.textLabel.text = @"cell";
[[cell
rac_prepareForReuseSignal]
subscribeNext:^(RACUnit * _Nullable x) {
NSLog(@"开始复用");
}];
return cell;
}
UITableViewHeaderFooterView+RACSignalSupport.h
#import <UIKit/UIKit.h> @class RACSignal<__covariant ValueType>;
@class RACUnit; NS_ASSUME_NONNULL_BEGIN // This category is only applicable to iOS >= 6.0.
@interface UITableViewHeaderFooterView (RACSignalSupport) @property (nonatomic, strong, readonly) RACSignal<RACUnit *> *rac_prepareForReuseSignal; @end NS_ASSUME_NONNULL_END
这个也是一样就不举例了
今天先看这些, 后面我们再继续
函数响应式编程及ReactiveObjC学习笔记 (四)的更多相关文章
- 函数响应式编程及ReactiveObjC学习笔记 (-)
最近无意间看到一个视频讲的ReactiveObjC, 觉得挺好用的 但听完后只是了解个大概. 在网上找了些文章, 有的写的比较易懂但看完还是没觉得自己能比较好的使用RAC, 有的甚至让我看不下去 这两 ...
- 函数响应式编程及ReactiveObjC学习笔记 (二)
之前我们初步认识了RAC的设计思路跟实现方式, 现在我们再来看看如果使用它以及它能帮我们做什么 One of the major advantages of RAC is that it provid ...
- 函数响应式编程及ReactiveObjC学习笔记 (三)
之前讲了RAC如何帮我们实现KVO / 代理 / 事件 / 通知 今天先不去分析它的核心代码, 我们先看看ReactiveObjC库里面一些特别的东西, 如果大家点开ReactiveObjC目录应该 ...
- 函数响应式编程(FRP)框架--ReactiveCocoa
由于工作原因,有段时间没更新博客了,甚是抱歉,只是,从今天開始我又活跃起来了,哈哈,于是决定每周更新一博.大家互相学习.交流. 今天呢.讨论一下关于ReactiveCocoa,这个採用函数响应式编程( ...
- 函数响应式编程(FRP)—基础概念篇
原文出处:http://ios.jobbole.com/86815/. 一函数响应式编程 说到函数响应式编程,就不得不提到函数式编程,他们俩有什么关系呢?今天我们就详细的解析一下他们的关系. 现在下面 ...
- RxJS入门之函数响应式编程
一.函数式编程 1.声明式(Declarativ) 和声明式相对应的编程⽅式叫做命令式编程(ImperativeProgramming),命令式编程也是最常见的⼀种编程⽅式. //命令式编程: fun ...
- 函数响应式编程(FRP)思想-Callback风格
序 ReactiveCocoa是IOS广为使用的技术框架,而ReactiveCocoa的核心思想就FRP.FRP不同于JAVA的object-oriented和AOP,FRP能让你的代码像数学一样简洁 ...
- ReactiveCocoa,最受欢迎的iOS函数响应式编程库(2.5版),没有之一!
简介 项目主页: ReactiveCocoa 实例下载: https://github.com/ios122/ios122 简评: 最受欢迎,最有价值的iOS响应式编程库,没有之一!iOS MVVM模 ...
- 函数响应式编程(FRP)从入门到”放弃”——基础概念篇
前言 研究ReactiveCocoa一段时间了,是时候总结一下学到的一些知识了. 一.函数响应式编程 说道函数响应式编程,就不得不提到函数式编程,它们俩到底有什么关系呢?今天我们就详细的解析一下他们的 ...
随机推荐
- 初识webpack——webpack四个基础概念
前面的话 webpack是当下最热门的前端资源模块化管理和打包工具.它可以将许多松散的模块按照依赖和规则打包成符合生产环境部署的前端资源.当webpack处理应用程序时,它会递归地构建一个依赖关系图表 ...
- 【Windows 10 应用开发】自定义快捷键
上一篇鸟文中,老周通过史无前例的代码向各位 demo 了访问键的用法(即 Alt + 某某).不过,大伙伴们一定会发现,访问键毕竟限制较大,不太灵活,也不好发挥,于是就需要自定义快捷键了. 其实,自定 ...
- Tomcat启动报错java.lang.UnsatisfiedLinkError
之前tomcat启动老是报错,虽然不影响项目的启动运行,但是有强迫症的程序员会心里不爽: 问题是由于本机安装的jdk版本与tomcat中使用的jdk版本不一致导致的. 后面我把原先tomcat启动环境 ...
- Perl格式化输出
Perl格式化输出 问题阐述 有时我们需要大量的重复数据,使用手工易于出错及比较繁琐.抓取特征,可以使用Perl脚本轻松搞定. 输出数据格式 主要特点 随机数生成 格式化输出 序列递增 Perl脚本 ...
- 【解决】使用compass watch xxx.scss 失败
原始日期:2016-01-25 16:49 在上一篇博客,我们终于安装好了compass,不过紧接着使用compass watch app.scss 结果失败,经过查询资料,是compass的版本问题 ...
- 基于Dubbo的分布式事务框架(LCN)
原文地址:http://原文地址:https://github.com/1991wangliang/transaction 基于Dubbo的分布式事务框架(LCN) 该框架依赖Redis/dubbo/ ...
- 使用dns批量管理普通主机名相关问题
1.dns配置 日常管理主机过程中,会有很多地方需要使用到主机名的,当主机非常多的时候,就不适合使用hosts来管理和同步的所有主机hosts了,这个时候就可以使用dns来管理主机名映射和变动 dns ...
- [leetcode-530-Minimum Absolute Difference in BST]
Given a binary search tree with non-negative values, find the minimum absolute difference between va ...
- Chapter 3. Video Coding Concepts
本章主要介绍一些有关视频编码的概念 时域模型(Temporal Model) 时域模型的作用是去除帧间冗余.如:将第二帧减去第一帧,得到的剩余信息,其能量会远小于第二帧本身. 基于块的运动估计和补偿 ...
- 【Android Developers Training】 11. 支持不同语言
注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...