您的位置:首页 > 产品设计 > UI/UE

UIButton使用RACCommand后setEnable方法失效的问题

2016-08-25 15:30 477 查看

问题描述

首先,我们知道RACCommand有如下初始化方法,可以传入enabledSignal,用于控制按钮的enable状态。

- (id)initWithEnabled:(RACSignal *)enabledSignal signalBlock:(RACSignal * (^)(id input))signalBlock;


但有时我们想自己控制按钮的enable状态。这个时候就可能出现标题所说的问题。标题为了表达意思,并不准确。真正的问题是RACCommand内部调用setEnable方法,可能会覆盖你手动设置的UIButton状态。无论你在初始化RACommand时是否传入enabledSignal,RACCommand内部都会控制UIButton的enable/disable状态。我们在外部也调用setEnable方法设置enable状态,可能会冲突。

举个栗子

RACCommand *sendCodeCommand = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) {
[sendCodeButton setEnabled:NO];
//TODO: Send code, count down and then setEnabled:YES
return [RACSignal empty];
}];
sendCodeButton.rac_command = sendCodeCommand;


上面这段代码,我们的预期是点击发送验证码按钮之后disable掉按钮,进行倒计时,倒计时结束重新enable按钮。事实上,我们点击按钮之后,按钮并没有被disable掉。

简单解决方案

放弃手动控制按钮的enable状态,在初始化RACCommand时候传入enabledSignal,让RACCommand统一负责。不过使用Signal控制enable状态需要额外的转换操作,要费些事。

RACCommand内部是如何控制UIButton enable状态的

如果我们知道RACCommand内部是如何控制enable状态的,我们就可以避免冲突。

我们在UIButton+RACCommandSupport.m可以找到下面这条语句:

disposable = [command.enabled setKeyPath:@keypath(self.enabled) onObject:self];


这条语句将UIButton的enable属性与RACCommand中的enabled信号绑定起来了,那么我们就来看看RACCommand中的enabled信号的如何创建的,如何变化的。

我们在RACCommand.m中找到下面这段代码:

if (enabledSignal == nil) {
enabledSignal = [RACSignal return:@YES];
} else {
enabledSignal = [[[enabledSignal
startWith:@YES]
takeUntil:self.rac_willDeallocSignal]
replayLast];
}
_immediateEnabled = [[RACSignal
combineLatest:@[ enabledSignal, moreExecutionsAllowed ]]
and];

_enabled = [[[[[self.immediateEnabled
take:1]
concat:[[self.immediateEnabled skip:1] deliverOn:RACScheduler.mainThreadScheduler]]
distinctUntilChanged]
replayLast]
setNameWithFormat:@"%@ -enabled", self];


可以看到enabled信号受我们传入的enabled信号(不传,默认为
[RACSignal return:@YES]
)和内部的moreExecutionsAllowed信号影响。它们进行与操作。再看看moreExecutionsAllowed信号的定义。

RACSignal *immediateExecuting = [RACObserve(self, activeExecutionSignals) map:^(NSArray *activeSignals) {
return @(activeSignals.count > 0);
}];

RACSignal *moreExecutionsAllowed = [RACSignal
if:RACObserve(self, allowsConcurrentExecution)
then:[RACSignal return:@YES]
else:[immediateExecuting not]];


moreExecutionsAllowed信号受allowsConcurrentExecution属性和immediateExecuting信号影响。如果允许并发,moreExecutionsAllowed发出的是@YES;如果不允许并发,目前有未完成任务,发出@NO,否则发出@YES。这里的任务完成是指signalBlock创建出来的Signal已经complete了,signalBlock是在
- (id)initWithEnabled:(RACSignal *)enabledSignal signalBlock:(RACSignal * (^)(id input))signalBlock
方法中传入的。

回到我们上面举的栗子,在signalBlock中我们disable了sendCodeButton,但是返回了
[RACSignal empty]
[RACSignal empty]
会立即complete,activeExecutionSignals会remove掉这个signal,然后activeSignals.count变为0,immediateExecuting发出@NO,moreExecutionsAllowed发出@YES,最终enabled信号发出@YES,sendCodeButton被enable。我们刚才设置的状态被冲掉了。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
相关文章推荐