今天我们继续看其他的类别

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学习笔记 (四)的更多相关文章

  1. 函数响应式编程及ReactiveObjC学习笔记 (-)

    最近无意间看到一个视频讲的ReactiveObjC, 觉得挺好用的 但听完后只是了解个大概. 在网上找了些文章, 有的写的比较易懂但看完还是没觉得自己能比较好的使用RAC, 有的甚至让我看不下去 这两 ...

  2. 函数响应式编程及ReactiveObjC学习笔记 (二)

    之前我们初步认识了RAC的设计思路跟实现方式, 现在我们再来看看如果使用它以及它能帮我们做什么 One of the major advantages of RAC is that it provid ...

  3. 函数响应式编程及ReactiveObjC学习笔记 (三)

    之前讲了RAC如何帮我们实现KVO / 代理 / 事件 / 通知 今天先不去分析它的核心代码, 我们先看看ReactiveObjC库里面一些特别的东西,  如果大家点开ReactiveObjC目录应该 ...

  4. 函数响应式编程(FRP)框架--ReactiveCocoa

    由于工作原因,有段时间没更新博客了,甚是抱歉,只是,从今天開始我又活跃起来了,哈哈,于是决定每周更新一博.大家互相学习.交流. 今天呢.讨论一下关于ReactiveCocoa,这个採用函数响应式编程( ...

  5. 函数响应式编程(FRP)—基础概念篇

    原文出处:http://ios.jobbole.com/86815/. 一函数响应式编程 说到函数响应式编程,就不得不提到函数式编程,他们俩有什么关系呢?今天我们就详细的解析一下他们的关系. 现在下面 ...

  6. RxJS入门之函数响应式编程

    一.函数式编程 1.声明式(Declarativ) 和声明式相对应的编程⽅式叫做命令式编程(ImperativeProgramming),命令式编程也是最常见的⼀种编程⽅式. //命令式编程: fun ...

  7. 函数响应式编程(FRP)思想-Callback风格

    序 ReactiveCocoa是IOS广为使用的技术框架,而ReactiveCocoa的核心思想就FRP.FRP不同于JAVA的object-oriented和AOP,FRP能让你的代码像数学一样简洁 ...

  8. ReactiveCocoa,最受欢迎的iOS函数响应式编程库(2.5版),没有之一!

    简介 项目主页: ReactiveCocoa 实例下载: https://github.com/ios122/ios122 简评: 最受欢迎,最有价值的iOS响应式编程库,没有之一!iOS MVVM模 ...

  9. 函数响应式编程(FRP)从入门到”放弃”——基础概念篇

    前言 研究ReactiveCocoa一段时间了,是时候总结一下学到的一些知识了. 一.函数响应式编程 说道函数响应式编程,就不得不提到函数式编程,它们俩到底有什么关系呢?今天我们就详细的解析一下他们的 ...

随机推荐

  1. oracle查询第一篇

    第一个小知识点: clear 在oracle中也可以用以清除屏幕上的内容 第二个小知识点: 在一个表中插入自身的查询结果 insert into my_table (id,name,age) sele ...

  2. Spring学习(7)--- @Required注解

    @Required注解是用于bean属性的setter方法 这个注解仅仅表示,受影响的bean属性必须在配置时被填充,通过在bean定义胡通过自动装配一个明确的属性值 package com.mypa ...

  3. 关系数据标准语言SQL之数据查询

    数据查询是数据库的核心操作.SQL提供了SELECT语句进行数据查询,该语句具有灵活的使用方式和丰富的功能. 其一般格式为 select [all | distinct]<目标表达式>[, ...

  4. javascript对象(1)

    今天说面向对象,嗯,不是那个对象,是这个对象. 接下来就开始今天的内容: 什么是面向对象: 就是把数据及数据的操作方法放在一起,作为一个相互依存的整体----对象.对同类对象抽象出其共性,形成类. 类 ...

  5. java hascode

    有部分代码如下: Cat cat=new Cat("Kitty",2);system.out.println(cat):问题:输出什么? 调用并执行toString()方法,两种情 ...

  6. Akka(10): 分布式运算:集群-Cluster

    Akka-Cluster可以在一部物理机或一组网络连接的服务器上搭建部署.用Akka开发同一版本的分布式程序可以在任何硬件环境中运行,这样我们就可以确定以Akka分布式程序作为标准的编程方式了. 在上 ...

  7. 九度OJ:1002-Grading

    时间限制:1 秒内存限制:32 兆特殊判题:否提交:24102解决:6126 题目描述: Grading hundreds of thousands of Graduate Entrance Exam ...

  8. Linux系统vi模式下显示行号

    在命令模式下输入:set nu或者:set number都可以为vi设置行号,如果要取消的话,则输入:set nonu行号的设置是vi的环境设置,不会影响文本的内容.

  9. linux服务器load的含义

    Linux的Load(系统负载),是一个让新手不太容易了解的概念.load的就是一定时间内计算机有多少个active_tasks,也就是说是计算机的任务执行队列的长度,cpu计算的队列. top/up ...

  10. [leetcode-543-Diameter of Binary Tree]

    Given a binary tree, you need to compute the length of the diameter of the tree. The diameter of a b ...